refactor: remove utils pkg
BREAKING CHANGE: utils package has been removed in favor of specific new packages (http, crypto, strings)
This commit is contained in:
parent
251c476e17
commit
0ab5ea5a57
40 changed files with 131 additions and 126 deletions
|
@ -12,8 +12,8 @@ import (
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/client/rp"
|
"github.com/caos/oidc/pkg/client/rp"
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -30,7 +30,7 @@ func main() {
|
||||||
scopes := strings.Split(os.Getenv("SCOPES"), " ")
|
scopes := strings.Split(os.Getenv("SCOPES"), " ")
|
||||||
|
|
||||||
redirectURI := fmt.Sprintf("http://localhost:%v%v", port, callbackPath)
|
redirectURI := fmt.Sprintf("http://localhost:%v%v", port, callbackPath)
|
||||||
cookieHandler := utils.NewCookieHandler(key, key, utils.WithUnsecure())
|
cookieHandler := httphelper.NewCookieHandler(key, key, httphelper.WithUnsecure())
|
||||||
|
|
||||||
options := []rp.Option{
|
options := []rp.Option{
|
||||||
rp.WithCookieHandler(cookieHandler),
|
rp.WithCookieHandler(cookieHandler),
|
||||||
|
|
|
@ -12,7 +12,7 @@ import (
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/client/rp"
|
"github.com/caos/oidc/pkg/client/rp"
|
||||||
"github.com/caos/oidc/pkg/client/rp/cli"
|
"github.com/caos/oidc/pkg/client/rp/cli"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -34,7 +34,7 @@ func main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
cookieHandler := utils.NewCookieHandler(key, key, utils.WithUnsecure())
|
cookieHandler := http.NewCookieHandler(key, key, http.WithUnsecure())
|
||||||
relyingParty, err := rp.NewRelyingPartyOAuth(rpConfig, rp.WithCookieHandler(cookieHandler))
|
relyingParty, err := rp.NewRelyingPartyOAuth(rpConfig, rp.WithCookieHandler(cookieHandler))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("error creating relaying party: %v", err)
|
fmt.Printf("error creating relaying party: %v", err)
|
||||||
|
|
|
@ -10,12 +10,13 @@ import (
|
||||||
"golang.org/x/oauth2"
|
"golang.org/x/oauth2"
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
|
"github.com/caos/oidc/pkg/crypto"
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
Encoder = func() utils.Encoder {
|
Encoder = func() httphelper.Encoder {
|
||||||
e := schema.NewEncoder()
|
e := schema.NewEncoder()
|
||||||
e.RegisterEncoder(oidc.SpaceDelimitedArray{}, func(value reflect.Value) string {
|
e.RegisterEncoder(oidc.SpaceDelimitedArray{}, func(value reflect.Value) string {
|
||||||
return value.Interface().(oidc.SpaceDelimitedArray).Encode()
|
return value.Interface().(oidc.SpaceDelimitedArray).Encode()
|
||||||
|
@ -32,7 +33,7 @@ func Discover(issuer string, httpClient *http.Client) (*oidc.DiscoveryConfigurat
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
discoveryConfig := new(oidc.DiscoveryConfiguration)
|
discoveryConfig := new(oidc.DiscoveryConfiguration)
|
||||||
err = utils.HttpRequest(httpClient, req, &discoveryConfig)
|
err = httphelper.HttpRequest(httpClient, req, &discoveryConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -52,12 +53,12 @@ func CallTokenEndpoint(request interface{}, caller tokenEndpointCaller) (newToke
|
||||||
}
|
}
|
||||||
|
|
||||||
func callTokenEndpoint(request interface{}, authFn interface{}, caller tokenEndpointCaller) (newToken *oauth2.Token, err error) {
|
func callTokenEndpoint(request interface{}, authFn interface{}, caller tokenEndpointCaller) (newToken *oauth2.Token, err error) {
|
||||||
req, err := utils.FormRequest(caller.TokenEndpoint(), request, Encoder, authFn)
|
req, err := httphelper.FormRequest(caller.TokenEndpoint(), request, Encoder, authFn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
tokenRes := new(oidc.AccessTokenResponse)
|
tokenRes := new(oidc.AccessTokenResponse)
|
||||||
if err := utils.HttpRequest(caller.HttpClient(), req, &tokenRes); err != nil {
|
if err := httphelper.HttpRequest(caller.HttpClient(), req, &tokenRes); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &oauth2.Token{
|
return &oauth2.Token{
|
||||||
|
@ -69,7 +70,7 @@ func callTokenEndpoint(request interface{}, authFn interface{}, caller tokenEndp
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSignerFromPrivateKeyByte(key []byte, keyID string) (jose.Signer, error) {
|
func NewSignerFromPrivateKeyByte(key []byte, keyID string) (jose.Signer, error) {
|
||||||
privateKey, err := utils.BytesToPrivateKey(key)
|
privateKey, err := crypto.BytesToPrivateKey(key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -83,7 +84,7 @@ func NewSignerFromPrivateKeyByte(key []byte, keyID string) (jose.Signer, error)
|
||||||
func SignedJWTProfileAssertion(clientID string, audience []string, expiration time.Duration, signer jose.Signer) (string, error) {
|
func SignedJWTProfileAssertion(clientID string, audience []string, expiration time.Duration, signer jose.Signer) (string, error) {
|
||||||
iat := time.Now()
|
iat := time.Now()
|
||||||
exp := iat.Add(expiration)
|
exp := iat.Add(expiration)
|
||||||
return utils.Sign(&oidc.JWTTokenRequest{
|
return crypto.Sign(&oidc.JWTTokenRequest{
|
||||||
Issuer: clientID,
|
Issuer: clientID,
|
||||||
Subject: clientID,
|
Subject: clientID,
|
||||||
Audience: audience,
|
Audience: audience,
|
||||||
|
|
|
@ -5,8 +5,8 @@ import (
|
||||||
|
|
||||||
"golang.org/x/oauth2"
|
"golang.org/x/oauth2"
|
||||||
|
|
||||||
|
"github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
//JWTProfileExchange handles the oauth2 jwt profile exchange
|
//JWTProfileExchange handles the oauth2 jwt profile exchange
|
||||||
|
@ -21,7 +21,7 @@ func ClientAssertionCodeOptions(assertion string) []oauth2.AuthCodeOption {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ClientAssertionFormAuthorization(assertion string) utils.FormAuthorization {
|
func ClientAssertionFormAuthorization(assertion string) http.FormAuthorization {
|
||||||
return func(values url.Values) {
|
return func(values url.Values) {
|
||||||
values.Set("client_assertion", assertion)
|
values.Set("client_assertion", assertion)
|
||||||
values.Set("client_assertion_type", oidc.ClientAssertionTypeJWTAssertion)
|
values.Set("client_assertion_type", oidc.ClientAssertionTypeJWTAssertion)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package cli
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
|
@ -5,8 +5,8 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/client/rp"
|
"github.com/caos/oidc/pkg/client/rp"
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -28,9 +28,9 @@ func CodeFlow(ctx context.Context, relyingParty rp.RelyingParty, callbackPath, p
|
||||||
http.Handle(loginPath, rp.AuthURLHandler(stateProvider, relyingParty))
|
http.Handle(loginPath, rp.AuthURLHandler(stateProvider, relyingParty))
|
||||||
http.Handle(callbackPath, rp.CodeExchangeHandler(callback, relyingParty))
|
http.Handle(callbackPath, rp.CodeExchangeHandler(callback, relyingParty))
|
||||||
|
|
||||||
utils.StartServer(codeflowCtx, ":"+port)
|
httphelper.StartServer(codeflowCtx, ":"+port)
|
||||||
|
|
||||||
utils.OpenBrowser("http://localhost:" + port + loginPath)
|
OpenBrowser("http://localhost:" + port + loginPath)
|
||||||
|
|
||||||
return <-tokenChan
|
return <-tokenChan
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,8 +5,8 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
//DelegationTokenRequest is an implementation of TokenExchangeRequest
|
//DelegationTokenRequest is an implementation of TokenExchangeRequest
|
||||||
//it exchanges a "urn:ietf:params:oauth:token-type:access_token" with an optional
|
//it exchanges an "urn:ietf:params:oauth:token-type:access_token" with an optional
|
||||||
//"urn:ietf:params:oauth:token-type:access_token" actor token for a
|
//"urn:ietf:params:oauth:token-type:access_token" actor token for an
|
||||||
//"urn:ietf:params:oauth:token-type:access_token" delegation token
|
//"urn:ietf:params:oauth:token-type:access_token" delegation token
|
||||||
func DelegationTokenRequest(subjectToken string, opts ...tokenexchange.TokenExchangeOption) *tokenexchange.TokenExchangeRequest {
|
func DelegationTokenRequest(subjectToken string, opts ...tokenexchange.TokenExchangeOption) *tokenexchange.TokenExchangeRequest {
|
||||||
return tokenexchange.NewTokenExchangeRequest(subjectToken, tokenexchange.AccessTokenType, opts...)
|
return tokenexchange.NewTokenExchangeRequest(subjectToken, tokenexchange.AccessTokenType, opts...)
|
||||||
|
|
|
@ -7,9 +7,9 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -207,7 +207,7 @@ func (r *remoteKeySet) fetchRemoteKeys(ctx context.Context) ([]jose.JSONWebKey,
|
||||||
}
|
}
|
||||||
|
|
||||||
keySet := new(jsonWebKeySet)
|
keySet := new(jsonWebKeySet)
|
||||||
if err = utils.HttpRequest(r.httpClient, req, keySet); err != nil {
|
if err = httphelper.HttpRequest(r.httpClient, req, keySet); err != nil {
|
||||||
return nil, fmt.Errorf("oidc: failed to get keys: %v", err)
|
return nil, fmt.Errorf("oidc: failed to get keys: %v", err)
|
||||||
}
|
}
|
||||||
return keySet.Keys, nil
|
return keySet.Keys, nil
|
||||||
|
|
|
@ -8,13 +8,13 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"golang.org/x/oauth2"
|
"golang.org/x/oauth2"
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/client"
|
"github.com/caos/oidc/pkg/client"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -39,7 +39,7 @@ type RelyingParty interface {
|
||||||
IsPKCE() bool
|
IsPKCE() bool
|
||||||
|
|
||||||
//CookieHandler returns a http cookie handler used for various state transfer cookies
|
//CookieHandler returns a http cookie handler used for various state transfer cookies
|
||||||
CookieHandler() *utils.CookieHandler
|
CookieHandler() *httphelper.CookieHandler
|
||||||
|
|
||||||
//HttpClient returns a http client used for calls to the openid provider, e.g. calling token endpoint
|
//HttpClient returns a http client used for calls to the openid provider, e.g. calling token endpoint
|
||||||
HttpClient() *http.Client
|
HttpClient() *http.Client
|
||||||
|
@ -76,7 +76,7 @@ type relyingParty struct {
|
||||||
pkce bool
|
pkce bool
|
||||||
|
|
||||||
httpClient *http.Client
|
httpClient *http.Client
|
||||||
cookieHandler *utils.CookieHandler
|
cookieHandler *httphelper.CookieHandler
|
||||||
|
|
||||||
errorHandler func(http.ResponseWriter, *http.Request, string, string, string)
|
errorHandler func(http.ResponseWriter, *http.Request, string, string, string)
|
||||||
idTokenVerifier IDTokenVerifier
|
idTokenVerifier IDTokenVerifier
|
||||||
|
@ -96,7 +96,7 @@ func (rp *relyingParty) IsPKCE() bool {
|
||||||
return rp.pkce
|
return rp.pkce
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rp *relyingParty) CookieHandler() *utils.CookieHandler {
|
func (rp *relyingParty) CookieHandler() *httphelper.CookieHandler {
|
||||||
return rp.cookieHandler
|
return rp.cookieHandler
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -136,7 +136,7 @@ func (rp *relyingParty) ErrorHandler() func(http.ResponseWriter, *http.Request,
|
||||||
func NewRelyingPartyOAuth(config *oauth2.Config, options ...Option) (RelyingParty, error) {
|
func NewRelyingPartyOAuth(config *oauth2.Config, options ...Option) (RelyingParty, error) {
|
||||||
rp := &relyingParty{
|
rp := &relyingParty{
|
||||||
oauthConfig: config,
|
oauthConfig: config,
|
||||||
httpClient: utils.DefaultHTTPClient,
|
httpClient: httphelper.DefaultHTTPClient,
|
||||||
oauth2Only: true,
|
oauth2Only: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -161,7 +161,7 @@ func NewRelyingPartyOIDC(issuer, clientID, clientSecret, redirectURI string, sco
|
||||||
RedirectURL: redirectURI,
|
RedirectURL: redirectURI,
|
||||||
Scopes: scopes,
|
Scopes: scopes,
|
||||||
},
|
},
|
||||||
httpClient: utils.DefaultHTTPClient,
|
httpClient: httphelper.DefaultHTTPClient,
|
||||||
oauth2Only: false,
|
oauth2Only: false,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -185,7 +185,7 @@ func NewRelyingPartyOIDC(issuer, clientID, clientSecret, redirectURI string, sco
|
||||||
type Option func(*relyingParty) error
|
type Option func(*relyingParty) error
|
||||||
|
|
||||||
//WithCookieHandler set a `CookieHandler` for securing the various redirects
|
//WithCookieHandler set a `CookieHandler` for securing the various redirects
|
||||||
func WithCookieHandler(cookieHandler *utils.CookieHandler) Option {
|
func WithCookieHandler(cookieHandler *httphelper.CookieHandler) Option {
|
||||||
return func(rp *relyingParty) error {
|
return func(rp *relyingParty) error {
|
||||||
rp.cookieHandler = cookieHandler
|
rp.cookieHandler = cookieHandler
|
||||||
return nil
|
return nil
|
||||||
|
@ -195,7 +195,7 @@ func WithCookieHandler(cookieHandler *utils.CookieHandler) Option {
|
||||||
//WithPKCE sets the RP to use PKCE (oauth2 code challenge)
|
//WithPKCE sets the RP to use PKCE (oauth2 code challenge)
|
||||||
//it also sets a `CookieHandler` for securing the various redirects
|
//it also sets a `CookieHandler` for securing the various redirects
|
||||||
//and exchanging the code challenge
|
//and exchanging the code challenge
|
||||||
func WithPKCE(cookieHandler *utils.CookieHandler) Option {
|
func WithPKCE(cookieHandler *httphelper.CookieHandler) Option {
|
||||||
return func(rp *relyingParty) error {
|
return func(rp *relyingParty) error {
|
||||||
rp.pkce = true
|
rp.pkce = true
|
||||||
rp.cookieHandler = cookieHandler
|
rp.cookieHandler = cookieHandler
|
||||||
|
@ -246,7 +246,7 @@ func Discover(issuer string, httpClient *http.Client) (Endpoints, error) {
|
||||||
return Endpoints{}, err
|
return Endpoints{}, err
|
||||||
}
|
}
|
||||||
discoveryConfig := new(oidc.DiscoveryConfiguration)
|
discoveryConfig := new(oidc.DiscoveryConfiguration)
|
||||||
err = utils.HttpRequest(httpClient, req, &discoveryConfig)
|
err = httphelper.HttpRequest(httpClient, req, &discoveryConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return Endpoints{}, err
|
return Endpoints{}, err
|
||||||
}
|
}
|
||||||
|
@ -395,7 +395,7 @@ func Userinfo(token, tokenType, subject string, rp RelyingParty) (oidc.UserInfo,
|
||||||
}
|
}
|
||||||
req.Header.Set("authorization", tokenType+" "+token)
|
req.Header.Set("authorization", tokenType+" "+token)
|
||||||
userinfo := oidc.NewUserInfo()
|
userinfo := oidc.NewUserInfo()
|
||||||
if err := utils.HttpRequest(rp.HttpClient(), req, &userinfo); err != nil {
|
if err := httphelper.HttpRequest(rp.HttpClient(), req, &userinfo); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if userinfo.GetSubject() != subject {
|
if userinfo.GetSubject() != subject {
|
||||||
|
|
|
@ -7,8 +7,8 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/client"
|
"github.com/caos/oidc/pkg/client"
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ResourceServer interface {
|
type ResourceServer interface {
|
||||||
|
@ -39,7 +39,7 @@ func (r *resourceServer) AuthFn() (interface{}, error) {
|
||||||
|
|
||||||
func NewResourceServerClientCredentials(issuer, clientID, clientSecret string, option ...Option) (ResourceServer, error) {
|
func NewResourceServerClientCredentials(issuer, clientID, clientSecret string, option ...Option) (ResourceServer, error) {
|
||||||
authorizer := func() (interface{}, error) {
|
authorizer := func() (interface{}, error) {
|
||||||
return utils.AuthorizeBasic(clientID, clientSecret), nil
|
return httphelper.AuthorizeBasic(clientID, clientSecret), nil
|
||||||
}
|
}
|
||||||
return newResourceServer(issuer, authorizer, option...)
|
return newResourceServer(issuer, authorizer, option...)
|
||||||
}
|
}
|
||||||
|
@ -61,7 +61,7 @@ func NewResourceServerJWTProfile(issuer, clientID, keyID string, key []byte, opt
|
||||||
func newResourceServer(issuer string, authorizer func() (interface{}, error), options ...Option) (*resourceServer, error) {
|
func newResourceServer(issuer string, authorizer func() (interface{}, error), options ...Option) (*resourceServer, error) {
|
||||||
rs := &resourceServer{
|
rs := &resourceServer{
|
||||||
issuer: issuer,
|
issuer: issuer,
|
||||||
httpClient: utils.DefaultHTTPClient,
|
httpClient: httphelper.DefaultHTTPClient,
|
||||||
}
|
}
|
||||||
for _, optFunc := range options {
|
for _, optFunc := range options {
|
||||||
optFunc(rs)
|
optFunc(rs)
|
||||||
|
@ -111,12 +111,12 @@ func Introspect(ctx context.Context, rp ResourceServer, token string) (oidc.Intr
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
req, err := utils.FormRequest(rp.IntrospectionURL(), &oidc.IntrospectionRequest{Token: token}, client.Encoder, authFn)
|
req, err := httphelper.FormRequest(rp.IntrospectionURL(), &oidc.IntrospectionRequest{Token: token}, client.Encoder, authFn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
resp := oidc.NewIntrospectionResponse()
|
resp := oidc.NewIntrospectionResponse()
|
||||||
if err := utils.HttpRequest(rp.HttpClient(), req, resp); err != nil {
|
if err := httphelper.HttpRequest(rp.HttpClient(), req, resp); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return resp, nil
|
return resp, nil
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package crypto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/aes"
|
"crypto/aes"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package crypto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/sha256"
|
"crypto/sha256"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package crypto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/rsa"
|
"crypto/rsa"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package crypto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
@ -14,7 +14,7 @@ import (
|
||||||
|
|
||||||
var (
|
var (
|
||||||
DefaultHTTPClient = &http.Client{
|
DefaultHTTPClient = &http.Client{
|
||||||
Timeout: time.Duration(30 * time.Second),
|
Timeout: 30 * time.Second,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
|
@ -3,7 +3,7 @@ package oidc
|
||||||
import (
|
import (
|
||||||
"crypto/sha256"
|
"crypto/sha256"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/crypto"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -19,7 +19,7 @@ type CodeChallenge struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSHACodeChallenge(code string) string {
|
func NewSHACodeChallenge(code string) string {
|
||||||
return utils.HashString(sha256.New(), code, false)
|
return crypto.HashString(sha256.New(), code, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
func VerifyCodeChallenge(c *CodeChallenge, codeVerifier string) bool {
|
func VerifyCodeChallenge(c *CodeChallenge, codeVerifier string) bool {
|
||||||
|
|
|
@ -9,7 +9,8 @@ import (
|
||||||
"golang.org/x/oauth2"
|
"golang.org/x/oauth2"
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/crypto"
|
||||||
|
"github.com/caos/oidc/pkg/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -185,7 +186,7 @@ func (a *accessTokenClaims) MarshalJSON() ([]byte, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return utils.ConcatenateJSON(b, info)
|
return http.ConcatenateJSON(b, info)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *accessTokenClaims) UnmarshalJSON(data []byte) error {
|
func (a *accessTokenClaims) UnmarshalJSON(data []byte) error {
|
||||||
|
@ -372,7 +373,7 @@ func (t *idTokenClaims) MarshalJSON() ([]byte, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return utils.ConcatenateJSON(b, info)
|
return http.ConcatenateJSON(b, info)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *idTokenClaims) UnmarshalJSON(data []byte) error {
|
func (t *idTokenClaims) UnmarshalJSON(data []byte) error {
|
||||||
|
@ -569,12 +570,12 @@ func NewJWTProfileAssertion(userID, keyID string, audience []string, key []byte,
|
||||||
}
|
}
|
||||||
|
|
||||||
func ClaimHash(claim string, sigAlgorithm jose.SignatureAlgorithm) (string, error) {
|
func ClaimHash(claim string, sigAlgorithm jose.SignatureAlgorithm) (string, error) {
|
||||||
hash, err := utils.GetHashAlgorithm(sigAlgorithm)
|
hash, err := crypto.GetHashAlgorithm(sigAlgorithm)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
return utils.HashString(hash, claim, true), nil
|
return crypto.HashString(hash, claim, true), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func AppendClientIDToAudience(clientID string, audience []string) []string {
|
func AppendClientIDToAudience(clientID string, audience []string) []string {
|
||||||
|
@ -587,7 +588,7 @@ func AppendClientIDToAudience(clientID string, audience []string) []string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateJWTProfileToken(assertion JWTProfileAssertionClaims) (string, error) {
|
func GenerateJWTProfileToken(assertion JWTProfileAssertionClaims) (string, error) {
|
||||||
privateKey, err := utils.BytesToPrivateKey(assertion.GetPrivateKey())
|
privateKey, err := crypto.BytesToPrivateKey(assertion.GetPrivateKey())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ import (
|
||||||
|
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
str "github.com/caos/oidc/pkg/strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Claims interface {
|
type Claims interface {
|
||||||
|
@ -61,10 +61,10 @@ type Verifier interface {
|
||||||
type ACRVerifier func(string) error
|
type ACRVerifier func(string) error
|
||||||
|
|
||||||
//DefaultACRVerifier implements `ACRVerifier` returning an error
|
//DefaultACRVerifier implements `ACRVerifier` returning an error
|
||||||
//if non of the provided values matches the acr claim
|
//if none of the provided values matches the acr claim
|
||||||
func DefaultACRVerifier(possibleValues []string) ACRVerifier {
|
func DefaultACRVerifier(possibleValues []string) ACRVerifier {
|
||||||
return func(acr string) error {
|
return func(acr string) error {
|
||||||
if !utils.Contains(possibleValues, acr) {
|
if !str.Contains(possibleValues, acr) {
|
||||||
return fmt.Errorf("expected one of: %v, got: %q", possibleValues, acr)
|
return fmt.Errorf("expected one of: %v, got: %q", possibleValues, acr)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -103,7 +103,7 @@ func CheckIssuer(claims Claims, issuer string) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func CheckAudience(claims Claims, clientID string) error {
|
func CheckAudience(claims Claims, clientID string) error {
|
||||||
if !utils.Contains(claims.GetAudience(), clientID) {
|
if !str.Contains(claims.GetAudience(), clientID) {
|
||||||
return fmt.Errorf("%w: Audience must contain client_id %q", ErrAudience, clientID)
|
return fmt.Errorf("%w: Audience must contain client_id %q", ErrAudience, clientID)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -138,7 +138,7 @@ func CheckSignature(ctx context.Context, token string, payload []byte, claims Cl
|
||||||
if len(supportedSigAlgs) == 0 {
|
if len(supportedSigAlgs) == 0 {
|
||||||
supportedSigAlgs = []string{"RS256"}
|
supportedSigAlgs = []string{"RS256"}
|
||||||
}
|
}
|
||||||
if !utils.Contains(supportedSigAlgs, sig.Header.Algorithm) {
|
if !str.Contains(supportedSigAlgs, sig.Header.Algorithm) {
|
||||||
return fmt.Errorf("%w: id token signed with unsupported algorithm, expected %q got %q", ErrSignatureUnsupportedAlg, supportedSigAlgs, sig.Header.Algorithm)
|
return fmt.Errorf("%w: id token signed with unsupported algorithm, expected %q got %q", ErrSignatureUnsupportedAlg, supportedSigAlgs, sig.Header.Algorithm)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,8 +11,9 @@ import (
|
||||||
|
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
str "github.com/caos/oidc/pkg/strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
type AuthRequest interface {
|
type AuthRequest interface {
|
||||||
|
@ -34,8 +35,8 @@ type AuthRequest interface {
|
||||||
|
|
||||||
type Authorizer interface {
|
type Authorizer interface {
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Encoder() utils.Encoder
|
Encoder() httphelper.Encoder
|
||||||
Signer() Signer
|
Signer() Signer
|
||||||
IDTokenHintVerifier() IDTokenHintVerifier
|
IDTokenHintVerifier() IDTokenHintVerifier
|
||||||
Crypto() Crypto
|
Crypto() Crypto
|
||||||
|
@ -92,7 +93,7 @@ func Authorize(w http.ResponseWriter, r *http.Request, authorizer Authorizer) {
|
||||||
}
|
}
|
||||||
|
|
||||||
//ParseAuthorizeRequest parsed the http request into a oidc.AuthRequest
|
//ParseAuthorizeRequest parsed the http request into a oidc.AuthRequest
|
||||||
func ParseAuthorizeRequest(r *http.Request, decoder utils.Decoder) (*oidc.AuthRequest, error) {
|
func ParseAuthorizeRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.AuthRequest, error) {
|
||||||
err := r.ParseForm()
|
err := r.ParseForm()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, oidc.ErrInvalidRequest().WithDescription("cannot parse form").WithParent(err)
|
return nil, oidc.ErrInvalidRequest().WithDescription("cannot parse form").WithParent(err)
|
||||||
|
@ -182,7 +183,7 @@ func ValidateAuthReqRedirectURI(client Client, uri string, responseType oidc.Res
|
||||||
"Please ensure it is added to the request. If you have any questions, you may contact the administrator of the application.")
|
"Please ensure it is added to the request. If you have any questions, you may contact the administrator of the application.")
|
||||||
}
|
}
|
||||||
if strings.HasPrefix(uri, "https://") {
|
if strings.HasPrefix(uri, "https://") {
|
||||||
if !utils.Contains(client.RedirectURIs(), uri) {
|
if !str.Contains(client.RedirectURIs(), uri) {
|
||||||
return oidc.ErrInvalidRequestRedirectURI().
|
return oidc.ErrInvalidRequestRedirectURI().
|
||||||
WithDescription("The requested redirect_uri is missing in the client configuration. " +
|
WithDescription("The requested redirect_uri is missing in the client configuration. " +
|
||||||
"If you have any questions, you may contact the administrator of the application.")
|
"If you have any questions, you may contact the administrator of the application.")
|
||||||
|
@ -192,7 +193,7 @@ func ValidateAuthReqRedirectURI(client Client, uri string, responseType oidc.Res
|
||||||
if client.ApplicationType() == ApplicationTypeNative {
|
if client.ApplicationType() == ApplicationTypeNative {
|
||||||
return validateAuthReqRedirectURINative(client, uri, responseType)
|
return validateAuthReqRedirectURINative(client, uri, responseType)
|
||||||
}
|
}
|
||||||
if !utils.Contains(client.RedirectURIs(), uri) {
|
if !str.Contains(client.RedirectURIs(), uri) {
|
||||||
return oidc.ErrInvalidRequestRedirectURI().WithDescription("The requested redirect_uri is missing in the client configuration. " +
|
return oidc.ErrInvalidRequestRedirectURI().WithDescription("The requested redirect_uri is missing in the client configuration. " +
|
||||||
"If you have any questions, you may contact the administrator of the application.")
|
"If you have any questions, you may contact the administrator of the application.")
|
||||||
}
|
}
|
||||||
|
@ -214,7 +215,7 @@ func ValidateAuthReqRedirectURI(client Client, uri string, responseType oidc.Res
|
||||||
func validateAuthReqRedirectURINative(client Client, uri string, responseType oidc.ResponseType) error {
|
func validateAuthReqRedirectURINative(client Client, uri string, responseType oidc.ResponseType) error {
|
||||||
parsedURL, isLoopback := HTTPLoopbackOrLocalhost(uri)
|
parsedURL, isLoopback := HTTPLoopbackOrLocalhost(uri)
|
||||||
isCustomSchema := !strings.HasPrefix(uri, "http://")
|
isCustomSchema := !strings.HasPrefix(uri, "http://")
|
||||||
if utils.Contains(client.RedirectURIs(), uri) {
|
if str.Contains(client.RedirectURIs(), uri) {
|
||||||
if isLoopback || isCustomSchema {
|
if isLoopback || isCustomSchema {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -339,7 +340,7 @@ func AuthResponseToken(w http.ResponseWriter, r *http.Request, authReq AuthReque
|
||||||
AuthRequestError(w, r, authReq, err, authorizer.Encoder())
|
AuthRequestError(w, r, authReq, err, authorizer.Encoder())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
params, err := utils.URLEncodeResponse(resp, authorizer.Encoder())
|
params, err := httphelper.URLEncodeResponse(resp, authorizer.Encoder())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
AuthRequestError(w, r, authReq, err, authorizer.Encoder())
|
AuthRequestError(w, r, authReq, err, authorizer.Encoder())
|
||||||
return
|
return
|
||||||
|
|
|
@ -13,10 +13,10 @@ import (
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/op"
|
"github.com/caos/oidc/pkg/op"
|
||||||
"github.com/caos/oidc/pkg/op/mock"
|
"github.com/caos/oidc/pkg/op/mock"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -77,7 +77,7 @@ import (
|
||||||
func TestParseAuthorizeRequest(t *testing.T) {
|
func TestParseAuthorizeRequest(t *testing.T) {
|
||||||
type args struct {
|
type args struct {
|
||||||
r *http.Request
|
r *http.Request
|
||||||
decoder utils.Decoder
|
decoder httphelper.Decoder
|
||||||
}
|
}
|
||||||
type res struct {
|
type res struct {
|
||||||
want *oidc.AuthRequest
|
want *oidc.AuthRequest
|
||||||
|
@ -103,7 +103,7 @@ func TestParseAuthorizeRequest(t *testing.T) {
|
||||||
"decoding error",
|
"decoding error",
|
||||||
args{
|
args{
|
||||||
&http.Request{URL: &url.URL{RawQuery: "unknown=value"}},
|
&http.Request{URL: &url.URL{RawQuery: "unknown=value"}},
|
||||||
func() utils.Decoder {
|
func() httphelper.Decoder {
|
||||||
decoder := schema.NewDecoder()
|
decoder := schema.NewDecoder()
|
||||||
decoder.IgnoreUnknownKeys(false)
|
decoder.IgnoreUnknownKeys(false)
|
||||||
return decoder
|
return decoder
|
||||||
|
@ -118,7 +118,7 @@ func TestParseAuthorizeRequest(t *testing.T) {
|
||||||
"parsing ok",
|
"parsing ok",
|
||||||
args{
|
args{
|
||||||
&http.Request{URL: &url.URL{RawQuery: "scope=openid"}},
|
&http.Request{URL: &url.URL{RawQuery: "scope=openid"}},
|
||||||
func() utils.Decoder {
|
func() httphelper.Decoder {
|
||||||
decoder := schema.NewDecoder()
|
decoder := schema.NewDecoder()
|
||||||
decoder.IgnoreUnknownKeys(false)
|
decoder.IgnoreUnknownKeys(false)
|
||||||
return decoder
|
return decoder
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
package op
|
package op
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/crypto"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Crypto interface {
|
type Crypto interface {
|
||||||
|
@ -18,9 +18,9 @@ func NewAESCrypto(key [32]byte) Crypto {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *aesCrypto) Encrypt(s string) (string, error) {
|
func (c *aesCrypto) Encrypt(s string) (string, error) {
|
||||||
return utils.EncryptAES(s, c.key)
|
return crypto.EncryptAES(s, c.key)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *aesCrypto) Decrypt(s string) (string, error) {
|
func (c *aesCrypto) Decrypt(s string) (string, error) {
|
||||||
return utils.DecryptAES(s, c.key)
|
return crypto.DecryptAES(s, c.key)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,8 +3,8 @@ package op
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func discoveryHandler(c Configuration, s Signer) func(http.ResponseWriter, *http.Request) {
|
func discoveryHandler(c Configuration, s Signer) func(http.ResponseWriter, *http.Request) {
|
||||||
|
@ -14,7 +14,7 @@ func discoveryHandler(c Configuration, s Signer) func(http.ResponseWriter, *http
|
||||||
}
|
}
|
||||||
|
|
||||||
func Discover(w http.ResponseWriter, config *oidc.DiscoveryConfiguration) {
|
func Discover(w http.ResponseWriter, config *oidc.DiscoveryConfiguration) {
|
||||||
utils.MarshalJSON(w, config)
|
httphelper.MarshalJSON(w, config)
|
||||||
}
|
}
|
||||||
|
|
||||||
func CreateDiscoveryConfig(c Configuration, s Signer) *oidc.DiscoveryConfiguration {
|
func CreateDiscoveryConfig(c Configuration, s Signer) *oidc.DiscoveryConfiguration {
|
||||||
|
|
|
@ -3,8 +3,8 @@ package op
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type ErrAuthRequest interface {
|
type ErrAuthRequest interface {
|
||||||
|
@ -13,7 +13,7 @@ type ErrAuthRequest interface {
|
||||||
GetState() string
|
GetState() string
|
||||||
}
|
}
|
||||||
|
|
||||||
func AuthRequestError(w http.ResponseWriter, r *http.Request, authReq ErrAuthRequest, err error, encoder utils.Encoder) {
|
func AuthRequestError(w http.ResponseWriter, r *http.Request, authReq ErrAuthRequest, err error, encoder httphelper.Encoder) {
|
||||||
if authReq == nil {
|
if authReq == nil {
|
||||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
|
@ -24,7 +24,7 @@ func AuthRequestError(w http.ResponseWriter, r *http.Request, authReq ErrAuthReq
|
||||||
http.Error(w, e.Description, http.StatusBadRequest)
|
http.Error(w, e.Description, http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
params, err := utils.URLEncodeResponse(e, encoder)
|
params, err := httphelper.URLEncodeResponse(e, encoder)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(w, err.Error(), http.StatusBadRequest)
|
http.Error(w, err.Error(), http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
|
@ -45,5 +45,5 @@ func RequestError(w http.ResponseWriter, r *http.Request, err error) {
|
||||||
if e.ErrorType == oidc.InvalidClient {
|
if e.ErrorType == oidc.InvalidClient {
|
||||||
status = 401
|
status = 401
|
||||||
}
|
}
|
||||||
utils.MarshalJSONWithStatus(w, e, status)
|
httphelper.MarshalJSONWithStatus(w, e, status)
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
|
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
type KeyProvider interface {
|
type KeyProvider interface {
|
||||||
|
@ -22,8 +22,8 @@ func keysHandler(k KeyProvider) func(http.ResponseWriter, *http.Request) {
|
||||||
func Keys(w http.ResponseWriter, r *http.Request, k KeyProvider) {
|
func Keys(w http.ResponseWriter, r *http.Request, k KeyProvider) {
|
||||||
keySet, err := k.GetKeySet(r.Context())
|
keySet, err := k.GetKeySet(r.Context())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
utils.MarshalJSONWithStatus(w, err, http.StatusInternalServerError)
|
httphelper.MarshalJSONWithStatus(w, err, http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
utils.MarshalJSON(w, keySet)
|
httphelper.MarshalJSON(w, keySet)
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,11 +5,11 @@
|
||||||
package mock
|
package mock
|
||||||
|
|
||||||
import (
|
import (
|
||||||
reflect "reflect"
|
"reflect"
|
||||||
|
|
||||||
op "github.com/caos/oidc/pkg/op"
|
utils "github.com/caos/oidc/pkg/http"
|
||||||
utils "github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/op"
|
||||||
gomock "github.com/golang/mock/gomock"
|
"github.com/golang/mock/gomock"
|
||||||
)
|
)
|
||||||
|
|
||||||
// MockAuthorizer is a mock of Authorizer interface.
|
// MockAuthorizer is a mock of Authorizer interface.
|
||||||
|
|
10
pkg/op/op.go
10
pkg/op/op.go
|
@ -12,8 +12,8 @@ import (
|
||||||
"golang.org/x/text/language"
|
"golang.org/x/text/language"
|
||||||
"gopkg.in/square/go-jose.v2"
|
"gopkg.in/square/go-jose.v2"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -41,8 +41,8 @@ var (
|
||||||
type OpenIDProvider interface {
|
type OpenIDProvider interface {
|
||||||
Configuration
|
Configuration
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Encoder() utils.Encoder
|
Encoder() httphelper.Encoder
|
||||||
IDTokenHintVerifier() IDTokenHintVerifier
|
IDTokenHintVerifier() IDTokenHintVerifier
|
||||||
AccessTokenVerifier() AccessTokenVerifier
|
AccessTokenVerifier() AccessTokenVerifier
|
||||||
Crypto() Crypto
|
Crypto() Crypto
|
||||||
|
@ -211,11 +211,11 @@ func (o *openidProvider) Storage() Storage {
|
||||||
return o.storage
|
return o.storage
|
||||||
}
|
}
|
||||||
|
|
||||||
func (o *openidProvider) Decoder() utils.Decoder {
|
func (o *openidProvider) Decoder() httphelper.Decoder {
|
||||||
return o.decoder
|
return o.decoder
|
||||||
}
|
}
|
||||||
|
|
||||||
func (o *openidProvider) Encoder() utils.Encoder {
|
func (o *openidProvider) Encoder() httphelper.Encoder {
|
||||||
return o.encoder
|
return o.encoder
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/caos/oidc/pkg/utils"
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ProbesFn func(context.Context) error
|
type ProbesFn func(context.Context) error
|
||||||
|
@ -49,7 +49,7 @@ func ReadyStorage(s Storage) ProbesFn {
|
||||||
}
|
}
|
||||||
|
|
||||||
func ok(w http.ResponseWriter) {
|
func ok(w http.ResponseWriter) {
|
||||||
utils.MarshalJSON(w, status{"ok"})
|
httphelper.MarshalJSON(w, status{"ok"})
|
||||||
}
|
}
|
||||||
|
|
||||||
type status struct {
|
type status struct {
|
||||||
|
|
|
@ -4,12 +4,12 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type SessionEnder interface {
|
type SessionEnder interface {
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
IDTokenHintVerifier() IDTokenHintVerifier
|
IDTokenHintVerifier() IDTokenHintVerifier
|
||||||
DefaultLogoutRedirectURI() string
|
DefaultLogoutRedirectURI() string
|
||||||
|
@ -44,7 +44,7 @@ func EndSession(w http.ResponseWriter, r *http.Request, ender SessionEnder) {
|
||||||
http.Redirect(w, r, session.RedirectURI, http.StatusFound)
|
http.Redirect(w, r, session.RedirectURI, http.StatusFound)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseEndSessionRequest(r *http.Request, decoder utils.Decoder) (*oidc.EndSessionRequest, error) {
|
func ParseEndSessionRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.EndSessionRequest, error) {
|
||||||
err := r.ParseForm()
|
err := r.ParseForm()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
return nil, oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
||||||
|
|
|
@ -4,8 +4,9 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/caos/oidc/pkg/crypto"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
type TokenCreator interface {
|
type TokenCreator interface {
|
||||||
|
@ -64,7 +65,7 @@ func createTokens(ctx context.Context, tokenRequest TokenRequest, storage Storag
|
||||||
func needsRefreshToken(tokenRequest TokenRequest, client Client) bool {
|
func needsRefreshToken(tokenRequest TokenRequest, client Client) bool {
|
||||||
switch req := tokenRequest.(type) {
|
switch req := tokenRequest.(type) {
|
||||||
case AuthRequest:
|
case AuthRequest:
|
||||||
return utils.Contains(req.GetScopes(), oidc.ScopeOfflineAccess) && req.GetResponseType() == oidc.ResponseTypeCode && ValidateGrantType(client, oidc.GrantTypeRefreshToken)
|
return strings.Contains(req.GetScopes(), oidc.ScopeOfflineAccess) && req.GetResponseType() == oidc.ResponseTypeCode && ValidateGrantType(client, oidc.GrantTypeRefreshToken)
|
||||||
case RefreshTokenRequest:
|
case RefreshTokenRequest:
|
||||||
return true
|
return true
|
||||||
default:
|
default:
|
||||||
|
@ -104,7 +105,7 @@ func CreateJWT(ctx context.Context, issuer string, tokenRequest TokenRequest, ex
|
||||||
}
|
}
|
||||||
claims.SetPrivateClaims(privateClaims)
|
claims.SetPrivateClaims(privateClaims)
|
||||||
}
|
}
|
||||||
return utils.Sign(claims, signer.Signer())
|
return crypto.Sign(claims, signer.Signer())
|
||||||
}
|
}
|
||||||
|
|
||||||
type IDTokenRequest interface {
|
type IDTokenRequest interface {
|
||||||
|
@ -151,7 +152,7 @@ func CreateIDToken(ctx context.Context, issuer string, request IDTokenRequest, v
|
||||||
claims.SetCodeHash(codeHash)
|
claims.SetCodeHash(codeHash)
|
||||||
}
|
}
|
||||||
|
|
||||||
return utils.Sign(claims, signer.Signer())
|
return crypto.Sign(claims, signer.Signer())
|
||||||
}
|
}
|
||||||
|
|
||||||
func removeUserinfoScopes(scopes []string) []string {
|
func removeUserinfoScopes(scopes []string) []string {
|
||||||
|
|
|
@ -4,8 +4,8 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
//CodeExchange handles the OAuth 2.0 authorization_code grant, including
|
//CodeExchange handles the OAuth 2.0 authorization_code grant, including
|
||||||
|
@ -29,11 +29,11 @@ func CodeExchange(w http.ResponseWriter, r *http.Request, exchanger Exchanger) {
|
||||||
RequestError(w, r, err)
|
RequestError(w, r, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
utils.MarshalJSON(w, resp)
|
httphelper.MarshalJSON(w, resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
//ParseAccessTokenRequest parsed the http request into a oidc.AccessTokenRequest
|
//ParseAccessTokenRequest parsed the http request into a oidc.AccessTokenRequest
|
||||||
func ParseAccessTokenRequest(r *http.Request, decoder utils.Decoder) (*oidc.AccessTokenRequest, error) {
|
func ParseAccessTokenRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.AccessTokenRequest, error) {
|
||||||
request := new(oidc.AccessTokenRequest)
|
request := new(oidc.AccessTokenRequest)
|
||||||
err := ParseAuthenticatedTokenRequest(r, decoder, request)
|
err := ParseAuthenticatedTokenRequest(r, decoder, request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -5,12 +5,12 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Introspector interface {
|
type Introspector interface {
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Crypto() Crypto
|
Crypto() Crypto
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
AccessTokenVerifier() AccessTokenVerifier
|
AccessTokenVerifier() AccessTokenVerifier
|
||||||
|
@ -36,16 +36,16 @@ func Introspect(w http.ResponseWriter, r *http.Request, introspector Introspecto
|
||||||
}
|
}
|
||||||
tokenID, subject, ok := getTokenIDAndSubject(r.Context(), introspector, token)
|
tokenID, subject, ok := getTokenIDAndSubject(r.Context(), introspector, token)
|
||||||
if !ok {
|
if !ok {
|
||||||
utils.MarshalJSON(w, response)
|
httphelper.MarshalJSON(w, response)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
err = introspector.Storage().SetIntrospectionFromToken(r.Context(), response, tokenID, subject, clientID)
|
err = introspector.Storage().SetIntrospectionFromToken(r.Context(), response, tokenID, subject, clientID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
utils.MarshalJSON(w, response)
|
httphelper.MarshalJSON(w, response)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
response.SetActive(true)
|
response.SetActive(true)
|
||||||
utils.MarshalJSON(w, response)
|
httphelper.MarshalJSON(w, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseTokenIntrospectionRequest(r *http.Request, introspector Introspector) (token, clientID string, err error) {
|
func ParseTokenIntrospectionRequest(r *http.Request, introspector Introspector) (token, clientID string, err error) {
|
||||||
|
|
|
@ -5,8 +5,8 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type JWTAuthorizationGrantExchanger interface {
|
type JWTAuthorizationGrantExchanger interface {
|
||||||
|
@ -37,10 +37,10 @@ func JWTProfile(w http.ResponseWriter, r *http.Request, exchanger JWTAuthorizati
|
||||||
RequestError(w, r, err)
|
RequestError(w, r, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
utils.MarshalJSON(w, resp)
|
httphelper.MarshalJSON(w, resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseJWTProfileGrantRequest(r *http.Request, decoder utils.Decoder) (*oidc.JWTProfileGrantRequest, error) {
|
func ParseJWTProfileGrantRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.JWTProfileGrantRequest, error) {
|
||||||
err := r.ParseForm()
|
err := r.ParseForm()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
return nil, oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
||||||
|
@ -74,6 +74,6 @@ func CreateJWTTokenResponse(ctx context.Context, tokenRequest TokenRequest, crea
|
||||||
//ParseJWTProfileRequest has been renamed to ParseJWTProfileGrantRequest
|
//ParseJWTProfileRequest has been renamed to ParseJWTProfileGrantRequest
|
||||||
//
|
//
|
||||||
//deprecated: use ParseJWTProfileGrantRequest
|
//deprecated: use ParseJWTProfileGrantRequest
|
||||||
func ParseJWTProfileRequest(r *http.Request, decoder utils.Decoder) (*oidc.JWTProfileGrantRequest, error) {
|
func ParseJWTProfileRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.JWTProfileGrantRequest, error) {
|
||||||
return ParseJWTProfileGrantRequest(r, decoder)
|
return ParseJWTProfileGrantRequest(r, decoder)
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,8 +6,9 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
"github.com/caos/oidc/pkg/strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
type RefreshTokenRequest interface {
|
type RefreshTokenRequest interface {
|
||||||
|
@ -37,11 +38,11 @@ func RefreshTokenExchange(w http.ResponseWriter, r *http.Request, exchanger Exch
|
||||||
RequestError(w, r, err)
|
RequestError(w, r, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
utils.MarshalJSON(w, resp)
|
httphelper.MarshalJSON(w, resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
//ParseRefreshTokenRequest parsed the http request into a oidc.RefreshTokenRequest
|
//ParseRefreshTokenRequest parsed the http request into a oidc.RefreshTokenRequest
|
||||||
func ParseRefreshTokenRequest(r *http.Request, decoder utils.Decoder) (*oidc.RefreshTokenRequest, error) {
|
func ParseRefreshTokenRequest(r *http.Request, decoder httphelper.Decoder) (*oidc.RefreshTokenRequest, error) {
|
||||||
request := new(oidc.RefreshTokenRequest)
|
request := new(oidc.RefreshTokenRequest)
|
||||||
err := ParseAuthenticatedTokenRequest(r, decoder, request)
|
err := ParseAuthenticatedTokenRequest(r, decoder, request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -77,7 +78,7 @@ func ValidateRefreshTokenScopes(requestedScopes []string, authRequest RefreshTok
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
for _, scope := range requestedScopes {
|
for _, scope := range requestedScopes {
|
||||||
if !utils.Contains(authRequest.GetScopes(), scope) {
|
if !strings.Contains(authRequest.GetScopes(), scope) {
|
||||||
return oidc.ErrInvalidScope()
|
return oidc.ErrInvalidScope()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,14 +5,14 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Exchanger interface {
|
type Exchanger interface {
|
||||||
Issuer() string
|
Issuer() string
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Signer() Signer
|
Signer() Signer
|
||||||
Crypto() Crypto
|
Crypto() Crypto
|
||||||
AuthMethodPostSupported() bool
|
AuthMethodPostSupported() bool
|
||||||
|
@ -61,7 +61,7 @@ type AuthenticatedTokenRequest interface {
|
||||||
|
|
||||||
//ParseAuthenticatedTokenRequest parses the client_id and client_secret from the HTTP request from either
|
//ParseAuthenticatedTokenRequest parses the client_id and client_secret from the HTTP request from either
|
||||||
//HTTP Basic Auth header or form body and sets them into the provided authenticatedTokenRequest interface
|
//HTTP Basic Auth header or form body and sets them into the provided authenticatedTokenRequest interface
|
||||||
func ParseAuthenticatedTokenRequest(r *http.Request, decoder utils.Decoder, request AuthenticatedTokenRequest) error {
|
func ParseAuthenticatedTokenRequest(r *http.Request, decoder httphelper.Decoder, request AuthenticatedTokenRequest) error {
|
||||||
err := r.ParseForm()
|
err := r.ParseForm()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
return oidc.ErrInvalidRequest().WithDescription("error parsing form").WithParent(err)
|
||||||
|
|
|
@ -6,12 +6,12 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
httphelper "github.com/caos/oidc/pkg/http"
|
||||||
"github.com/caos/oidc/pkg/oidc"
|
"github.com/caos/oidc/pkg/oidc"
|
||||||
"github.com/caos/oidc/pkg/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type UserinfoProvider interface {
|
type UserinfoProvider interface {
|
||||||
Decoder() utils.Decoder
|
Decoder() httphelper.Decoder
|
||||||
Crypto() Crypto
|
Crypto() Crypto
|
||||||
Storage() Storage
|
Storage() Storage
|
||||||
AccessTokenVerifier() AccessTokenVerifier
|
AccessTokenVerifier() AccessTokenVerifier
|
||||||
|
@ -37,13 +37,13 @@ func Userinfo(w http.ResponseWriter, r *http.Request, userinfoProvider UserinfoP
|
||||||
info := oidc.NewUserInfo()
|
info := oidc.NewUserInfo()
|
||||||
err = userinfoProvider.Storage().SetUserinfoFromToken(r.Context(), info, tokenID, subject, r.Header.Get("origin"))
|
err = userinfoProvider.Storage().SetUserinfoFromToken(r.Context(), info, tokenID, subject, r.Header.Get("origin"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
utils.MarshalJSONWithStatus(w, err, http.StatusForbidden)
|
httphelper.MarshalJSONWithStatus(w, err, http.StatusForbidden)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
utils.MarshalJSON(w, info)
|
httphelper.MarshalJSON(w, info)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseUserinfoRequest(r *http.Request, decoder utils.Decoder) (string, error) {
|
func ParseUserinfoRequest(r *http.Request, decoder httphelper.Decoder) (string, error) {
|
||||||
accessToken, err := getAccessToken(r)
|
accessToken, err := getAccessToken(r)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
return accessToken, nil
|
return accessToken, nil
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package strings
|
||||||
|
|
||||||
func Contains(list []string, needle string) bool {
|
func Contains(list []string, needle string) bool {
|
||||||
for _, item := range list {
|
for _, item := range list {
|
|
@ -1,4 +1,4 @@
|
||||||
package utils
|
package strings
|
||||||
|
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue