2021-06-15 16:27:30 +00:00
|
|
|
// Copyright 2020-2021 the Pinniped contributors. All Rights Reserved.
|
2020-12-02 01:18:32 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package openidconnect
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
2020-12-10 22:47:58 +00:00
|
|
|
"time"
|
2020-12-02 01:18:32 +00:00
|
|
|
|
|
|
|
"github.com/ory/fosite"
|
|
|
|
"github.com/ory/fosite/handler/openid"
|
|
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
|
|
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
|
|
|
|
|
|
|
|
"go.pinniped.dev/internal/constable"
|
|
|
|
"go.pinniped.dev/internal/crud"
|
|
|
|
"go.pinniped.dev/internal/fositestorage"
|
2021-06-15 16:27:30 +00:00
|
|
|
"go.pinniped.dev/internal/oidc/clientregistry"
|
2021-10-06 22:28:13 +00:00
|
|
|
"go.pinniped.dev/internal/psession"
|
2020-12-02 01:18:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-12-04 23:40:17 +00:00
|
|
|
TypeLabelValue = "oidc"
|
|
|
|
|
2020-12-02 01:18:32 +00:00
|
|
|
ErrInvalidOIDCRequestVersion = constable.Error("oidc request data has wrong version")
|
|
|
|
ErrInvalidOIDCRequestData = constable.Error("oidc request data must be present")
|
|
|
|
ErrMalformedAuthorizationCode = constable.Error("malformed authorization code")
|
|
|
|
|
2021-10-06 22:28:13 +00:00
|
|
|
oidcStorageVersion = "2"
|
2020-12-02 01:18:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var _ openid.OpenIDConnectRequestStorage = &openIDConnectRequestStorage{}
|
|
|
|
|
|
|
|
type openIDConnectRequestStorage struct {
|
|
|
|
storage crud.Storage
|
|
|
|
}
|
|
|
|
|
|
|
|
type session struct {
|
|
|
|
Request *fosite.Request `json:"request"`
|
|
|
|
Version string `json:"version"`
|
|
|
|
}
|
|
|
|
|
2020-12-10 22:47:58 +00:00
|
|
|
func New(secrets corev1client.SecretInterface, clock func() time.Time, sessionStorageLifetime time.Duration) openid.OpenIDConnectRequestStorage {
|
|
|
|
return &openIDConnectRequestStorage{storage: crud.New(TypeLabelValue, secrets, clock, sessionStorageLifetime)}
|
2020-12-02 01:18:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *openIDConnectRequestStorage) CreateOpenIDConnectSession(ctx context.Context, authcode string, requester fosite.Requester) error {
|
|
|
|
signature, err := getSignature(authcode)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
request, err := fositestorage.ValidateAndExtractAuthorizeRequest(requester)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-12-04 22:31:06 +00:00
|
|
|
_, err = a.storage.Create(ctx, signature, &session{Request: request, Version: oidcStorageVersion}, nil)
|
2020-12-02 01:18:32 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *openIDConnectRequestStorage) GetOpenIDConnectSession(ctx context.Context, authcode string, _ fosite.Requester) (fosite.Requester, error) {
|
|
|
|
signature, err := getSignature(authcode)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
session, _, err := a.getSession(ctx, signature)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return session.Request, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *openIDConnectRequestStorage) DeleteOpenIDConnectSession(ctx context.Context, authcode string) error {
|
|
|
|
signature, err := getSignature(authcode)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return a.storage.Delete(ctx, signature)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *openIDConnectRequestStorage) getSession(ctx context.Context, signature string) (*session, string, error) {
|
|
|
|
session := newValidEmptyOIDCSession()
|
|
|
|
rv, err := a.storage.Get(ctx, signature, session)
|
|
|
|
|
|
|
|
if errors.IsNotFound(err) {
|
2020-12-17 20:09:19 +00:00
|
|
|
return nil, "", fosite.ErrNotFound.WithWrap(err).WithDebug(err.Error())
|
2020-12-02 01:18:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", fmt.Errorf("failed to get oidc session for %s: %w", signature, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if version := session.Version; version != oidcStorageVersion {
|
|
|
|
return nil, "", fmt.Errorf("%w: oidc session for %s has version %s instead of %s",
|
|
|
|
ErrInvalidOIDCRequestVersion, signature, version, oidcStorageVersion)
|
|
|
|
}
|
|
|
|
|
|
|
|
if session.Request.ID == "" {
|
|
|
|
return nil, "", fmt.Errorf("malformed oidc session for %s: %w", signature, ErrInvalidOIDCRequestData)
|
|
|
|
}
|
|
|
|
|
|
|
|
return session, rv, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func newValidEmptyOIDCSession() *session {
|
|
|
|
return &session{
|
|
|
|
Request: &fosite.Request{
|
2021-06-15 16:27:30 +00:00
|
|
|
Client: &clientregistry.Client{},
|
2021-10-06 22:28:13 +00:00
|
|
|
Session: &psession.PinnipedSession{},
|
2020-12-02 01:18:32 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getSignature(authorizationCode string) (string, error) {
|
|
|
|
split := strings.Split(authorizationCode, ".")
|
|
|
|
|
|
|
|
if len(split) != 2 {
|
|
|
|
return "", ErrMalformedAuthorizationCode
|
|
|
|
}
|
|
|
|
|
|
|
|
return split[1], nil
|
|
|
|
}
|