2021-10-06 22:28:13 +00:00
|
|
|
// Copyright 2020-2021 the Pinniped contributors. All Rights Reserved.
|
2020-12-01 21:25:12 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
// Package token provides a handler for the OIDC token endpoint.
|
|
|
|
package token
|
|
|
|
|
|
|
|
import (
|
2021-10-13 19:31:20 +00:00
|
|
|
"context"
|
2020-12-01 21:25:12 +00:00
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/ory/fosite"
|
2021-10-13 19:31:20 +00:00
|
|
|
"github.com/ory/x/errorsx"
|
2020-12-01 21:25:12 +00:00
|
|
|
|
|
|
|
"go.pinniped.dev/internal/httputil/httperr"
|
2020-12-04 15:06:55 +00:00
|
|
|
"go.pinniped.dev/internal/oidc"
|
2021-10-13 19:31:20 +00:00
|
|
|
"go.pinniped.dev/internal/oidc/provider"
|
2020-12-01 21:25:12 +00:00
|
|
|
"go.pinniped.dev/internal/plog"
|
2021-10-06 22:28:13 +00:00
|
|
|
"go.pinniped.dev/internal/psession"
|
2020-12-01 21:25:12 +00:00
|
|
|
)
|
|
|
|
|
2021-10-13 19:31:20 +00:00
|
|
|
var (
|
|
|
|
errMissingUpstreamSessionInternalError = &fosite.RFC6749Error{
|
|
|
|
ErrorField: "error",
|
|
|
|
DescriptionField: "There was an internal server error.",
|
|
|
|
HintField: "Required upstream data not found in session.",
|
|
|
|
CodeField: http.StatusInternalServerError,
|
|
|
|
}
|
|
|
|
|
|
|
|
errUpstreamRefreshError = &fosite.RFC6749Error{
|
|
|
|
ErrorField: "error",
|
|
|
|
DescriptionField: "Error during upstream refresh.",
|
|
|
|
CodeField: http.StatusUnauthorized,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2020-12-01 21:25:12 +00:00
|
|
|
func NewHandler(
|
2021-10-13 19:31:20 +00:00
|
|
|
idpLister oidc.UpstreamIdentityProvidersLister,
|
2020-12-01 21:25:12 +00:00
|
|
|
oauthHelper fosite.OAuth2Provider,
|
|
|
|
) http.Handler {
|
|
|
|
return httperr.HandlerFunc(func(w http.ResponseWriter, r *http.Request) error {
|
2021-10-06 22:28:13 +00:00
|
|
|
session := psession.NewPinnipedSession()
|
|
|
|
accessRequest, err := oauthHelper.NewAccessRequest(r.Context(), r, session)
|
2020-12-01 21:25:12 +00:00
|
|
|
if err != nil {
|
2020-12-04 15:06:55 +00:00
|
|
|
plog.Info("token request error", oidc.FositeErrorForLog(err)...)
|
2020-12-01 21:25:12 +00:00
|
|
|
oauthHelper.WriteAccessError(w, accessRequest, err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-13 19:31:20 +00:00
|
|
|
// Check if we are performing a refresh grant.
|
|
|
|
if accessRequest.GetGrantTypes().ExactOne("refresh_token") {
|
|
|
|
// The above call to NewAccessRequest has loaded the session from storage into the accessRequest variable.
|
|
|
|
// The session, requested scopes, and requested audience from the original authorize request was retrieved
|
|
|
|
// from the Kube storage layer and added to the accessRequest. Additionally, the audience and scopes may
|
|
|
|
// have already been granted on the accessRequest.
|
|
|
|
err = upstreamRefresh(r.Context(), accessRequest, idpLister)
|
|
|
|
if err != nil {
|
|
|
|
plog.Info("upstream refresh error", oidc.FositeErrorForLog(err)...)
|
|
|
|
oauthHelper.WriteAccessError(w, accessRequest, err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-01 21:25:12 +00:00
|
|
|
accessResponse, err := oauthHelper.NewAccessResponse(r.Context(), accessRequest)
|
|
|
|
if err != nil {
|
2020-12-04 15:06:55 +00:00
|
|
|
plog.Info("token response error", oidc.FositeErrorForLog(err)...)
|
2020-12-01 21:25:12 +00:00
|
|
|
oauthHelper.WriteAccessError(w, accessRequest, err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
oauthHelper.WriteAccessResponse(w, accessRequest, accessResponse)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
2021-10-13 19:31:20 +00:00
|
|
|
|
|
|
|
func upstreamRefresh(ctx context.Context, accessRequest fosite.AccessRequester, providerCache oidc.UpstreamIdentityProvidersLister) error {
|
|
|
|
session := accessRequest.GetSession().(*psession.PinnipedSession)
|
|
|
|
customSessionData := session.Custom
|
|
|
|
if customSessionData == nil {
|
|
|
|
return errorsx.WithStack(errMissingUpstreamSessionInternalError)
|
|
|
|
}
|
|
|
|
providerName := customSessionData.ProviderName
|
|
|
|
providerUID := customSessionData.ProviderUID
|
|
|
|
if providerUID == "" || providerName == "" {
|
|
|
|
return errorsx.WithStack(errMissingUpstreamSessionInternalError)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch customSessionData.ProviderType {
|
|
|
|
case psession.ProviderTypeOIDC:
|
|
|
|
return upstreamOIDCRefresh(ctx, customSessionData, providerCache)
|
|
|
|
case psession.ProviderTypeLDAP:
|
|
|
|
// upstream refresh not yet implemented for LDAP, so do nothing
|
|
|
|
case psession.ProviderTypeActiveDirectory:
|
|
|
|
// upstream refresh not yet implemented for AD, so do nothing
|
|
|
|
default:
|
|
|
|
return errorsx.WithStack(errMissingUpstreamSessionInternalError)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func upstreamOIDCRefresh(ctx context.Context, s *psession.CustomSessionData, providerCache oidc.UpstreamIdentityProvidersLister) error {
|
|
|
|
if s.OIDC == nil || s.OIDC.UpstreamRefreshToken == "" {
|
|
|
|
return errorsx.WithStack(errMissingUpstreamSessionInternalError)
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := findOIDCProviderByNameAndValidateUID(s, providerCache)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
plog.Debug("attempting upstream refresh request",
|
|
|
|
"providerName", s.ProviderName, "providerType", s.ProviderType, "providerUID", s.ProviderUID)
|
|
|
|
|
|
|
|
refreshedTokens, err := p.PerformRefresh(ctx, s.OIDC.UpstreamRefreshToken)
|
|
|
|
if err != nil {
|
|
|
|
return errorsx.WithStack(errUpstreamRefreshError.WithHintf(
|
|
|
|
"Upstream refresh failed using provider %q of type %q.",
|
|
|
|
s.ProviderName, s.ProviderType).WithWrap(err))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Upstream refresh may or may not return a new ID token. From the spec:
|
|
|
|
// "the response body is the Token Response of Section 3.1.3.3 except that it might not contain an id_token."
|
|
|
|
// https://openid.net/specs/openid-connect-core-1_0.html#RefreshTokenResponse
|
|
|
|
_, hasIDTok := refreshedTokens.Extra("id_token").(string)
|
|
|
|
if hasIDTok {
|
|
|
|
// The spec is not 100% clear about whether an ID token from the refresh flow should include a nonce, and at
|
|
|
|
// least some providers do not include one, so we skip the nonce validation here (but not other validations).
|
|
|
|
_, err = p.ValidateToken(ctx, refreshedTokens, "")
|
|
|
|
if err != nil {
|
|
|
|
return errorsx.WithStack(errUpstreamRefreshError.WithHintf(
|
|
|
|
"Upstream refresh returned an invalid ID token using provider %q of type %q.",
|
|
|
|
s.ProviderName, s.ProviderType).WithWrap(err))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
plog.Debug("upstream refresh request did not return a new ID token",
|
|
|
|
"providerName", s.ProviderName, "providerType", s.ProviderType, "providerUID", s.ProviderUID)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Upstream refresh may or may not return a new refresh token. If we got a new refresh token, then update it in
|
|
|
|
// the user's session. If we did not get a new refresh token, then keep the old one in the session by avoiding
|
|
|
|
// overwriting the old one.
|
|
|
|
if refreshedTokens.RefreshToken != "" {
|
|
|
|
plog.Debug("upstream refresh request did not return a new refresh token",
|
|
|
|
"providerName", s.ProviderName, "providerType", s.ProviderType, "providerUID", s.ProviderUID)
|
|
|
|
s.OIDC.UpstreamRefreshToken = refreshedTokens.RefreshToken
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func findOIDCProviderByNameAndValidateUID(
|
|
|
|
s *psession.CustomSessionData,
|
|
|
|
providerCache oidc.UpstreamIdentityProvidersLister,
|
|
|
|
) (provider.UpstreamOIDCIdentityProviderI, error) {
|
|
|
|
for _, p := range providerCache.GetOIDCIdentityProviders() {
|
|
|
|
if p.GetName() == s.ProviderName {
|
|
|
|
if p.GetResourceUID() != s.ProviderUID {
|
|
|
|
return nil, errorsx.WithStack(errUpstreamRefreshError.WithHintf(
|
|
|
|
"Provider %q of type %q from upstream session data has changed its resource UID since authentication.",
|
|
|
|
s.ProviderName, s.ProviderType))
|
|
|
|
}
|
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errorsx.WithStack(errUpstreamRefreshError.
|
|
|
|
WithHintf("Provider %q of type %q from upstream session data was not found.", s.ProviderName, s.ProviderType))
|
|
|
|
}
|