server: CodeToken now does Cross-Client auth

This commit is contained in:
Bobby Rullo 2016-06-02 13:05:18 -07:00
parent 9b4740862c
commit e71c5086ba
8 changed files with 269 additions and 67 deletions

34
scope/scope.go Normal file
View file

@ -0,0 +1,34 @@
package scope
import "strings"
const (
// Scope prefix which indicates initiation of a cross-client authentication flow.
// See https://developers.google.com/identity/protocols/CrossClientAuth
ScopeGoogleCrossClient = "audience:server:client_id:"
)
type Scopes []string
func (s Scopes) OfflineAccess() bool {
return s.HasScope("offline_access")
}
func (s Scopes) HasScope(scope string) bool {
for _, curScope := range s {
if curScope == scope {
return true
}
}
return false
}
func (s Scopes) CrossClientIDs() []string {
clients := []string{}
for _, scope := range s {
if strings.HasPrefix(scope, ScopeGoogleCrossClient) {
clients = append(clients, scope[len(ScopeGoogleCrossClient):])
}
}
return clients
}

View file

@ -1,17 +1,22 @@
package server package server
import ( import (
"encoding/base64"
"fmt" "fmt"
"net/http" "net/http"
"net/http/httptest" "net/http/httptest"
"net/url" "net/url"
"sort"
"strings" "strings"
"testing" "testing"
"github.com/coreos/go-oidc/oidc" "github.com/coreos/go-oidc/oidc"
"github.com/kylelemons/godebug/pretty"
"github.com/coreos/dex/client" "github.com/coreos/dex/client"
clientmanager "github.com/coreos/dex/client/manager"
"github.com/coreos/dex/connector" "github.com/coreos/dex/connector"
"github.com/coreos/dex/scope"
) )
func makeCrossClientTestFixtures() (*testFixtures, error) { func makeCrossClientTestFixtures() (*testFixtures, error) {
@ -20,7 +25,6 @@ func makeCrossClientTestFixtures() (*testFixtures, error) {
return nil, fmt.Errorf("couldn't make test fixtures: %v", err) return nil, fmt.Errorf("couldn't make test fixtures: %v", err)
} }
creds := map[string]oidc.ClientCredentials{}
for _, cliData := range []struct { for _, cliData := range []struct {
id string id string
authorized []string authorized []string
@ -38,24 +42,22 @@ func makeCrossClientTestFixtures() (*testFixtures, error) {
u := url.URL{ u := url.URL{
Scheme: "https://", Scheme: "https://",
Path: cliData.id, Path: cliData.id,
Host: "auth.example.com", Host: cliData.id,
} }
cliCreds, err := f.clientRepo.New(client.Client{ cliCreds, err := f.clientManager.New(client.Client{
Credentials: oidc.ClientCredentials{ Credentials: oidc.ClientCredentials{
ID: cliData.id, ID: cliData.id,
}, },
Metadata: oidc.ClientMetadata{ Metadata: oidc.ClientMetadata{
RedirectURIs: []url.URL{u}, RedirectURIs: []url.URL{u},
}, },
}, &clientmanager.ClientOptions{
TrustedPeers: cliData.authorized,
}) })
if err != nil { if err != nil {
return nil, fmt.Errorf("Unexpected error creating clients: %v", err) return nil, fmt.Errorf("Unexpected error creating clients: %v", err)
} }
creds[cliData.id] = *cliCreds f.clientCreds[cliData.id] = *cliCreds
err = f.clientRepo.SetTrustedPeers(cliData.id, cliData.authorized)
if err != nil {
return nil, fmt.Errorf("Unexpected error setting cross-client authorizers: %v", err)
}
} }
return f, nil return f, nil
} }
@ -132,30 +134,30 @@ func TestHandleAuthCrossClient(t *testing.T) {
wantCode int wantCode int
}{ }{
{ {
scopes: []string{ScopeGoogleCrossClient + "client_a"}, scopes: []string{scope.ScopeGoogleCrossClient + "client_a"},
clientID: "client_b", clientID: "client_b",
wantCode: http.StatusBadRequest, wantCode: http.StatusBadRequest,
}, },
{ {
scopes: []string{ScopeGoogleCrossClient + "client_b"}, scopes: []string{scope.ScopeGoogleCrossClient + "client_b"},
clientID: "client_a", clientID: "client_a",
wantCode: http.StatusFound, wantCode: http.StatusFound,
}, },
{ {
scopes: []string{ScopeGoogleCrossClient + "client_b"}, scopes: []string{scope.ScopeGoogleCrossClient + "client_b"},
clientID: "client_a", clientID: "client_a",
wantCode: http.StatusFound, wantCode: http.StatusFound,
}, },
{ {
scopes: []string{ScopeGoogleCrossClient + "client_c"}, scopes: []string{scope.ScopeGoogleCrossClient + "client_c"},
clientID: "client_a", clientID: "client_a",
wantCode: http.StatusFound, wantCode: http.StatusFound,
}, },
{ {
// Two clients that client_a is authorized to mint tokens for. // Two clients that client_a is authorized to mint tokens for.
scopes: []string{ scopes: []string{
ScopeGoogleCrossClient + "client_c", scope.ScopeGoogleCrossClient + "client_c",
ScopeGoogleCrossClient + "client_b", scope.ScopeGoogleCrossClient + "client_b",
}, },
clientID: "client_a", clientID: "client_a",
wantCode: http.StatusFound, wantCode: http.StatusFound,
@ -163,8 +165,8 @@ func TestHandleAuthCrossClient(t *testing.T) {
{ {
// Two clients that client_a is authorized to mint tokens for. // Two clients that client_a is authorized to mint tokens for.
scopes: []string{ scopes: []string{
ScopeGoogleCrossClient + "client_c", scope.ScopeGoogleCrossClient + "client_c",
ScopeGoogleCrossClient + "client_a", scope.ScopeGoogleCrossClient + "client_a",
}, },
clientID: "client_b", clientID: "client_b",
wantCode: http.StatusBadRequest, wantCode: http.StatusBadRequest,
@ -200,3 +202,129 @@ func TestHandleAuthCrossClient(t *testing.T) {
} }
} }
func TestServerCodeTokenCrossClient(t *testing.T) {
f, err := makeCrossClientTestFixtures()
if err != nil {
t.Fatalf("Error creating test fixtures: %v", err)
}
sm := f.sessionManager
tests := []struct {
clientID string
offline bool
refreshToken string
crossClients []string
wantErr bool
wantAUD []string
wantAZP string
}{
// First test the non-cross-client cases, make sure they're undisturbed:
{
// No 'offline_access' in scope, should get empty refresh token.
clientID: testClientID,
refreshToken: "",
wantAUD: []string{testClientID},
},
{
// Have 'offline_access' in scope, should get non-empty refresh token.
clientID: testClientID,
offline: true,
refreshToken: fmt.Sprintf("1/%s", base64.URLEncoding.EncodeToString([]byte("refresh-1"))),
wantAUD: []string{testClientID},
},
// Now test cross-client cases:
{
clientID: "client_a",
crossClients: []string{"client_b"},
wantAUD: []string{"client_b"},
wantAZP: "client_a",
},
{
clientID: "client_a",
crossClients: []string{"client_b", "client_a"},
wantAUD: []string{"client_a", "client_b"},
wantAZP: "client_a",
},
}
for i, tt := range tests {
scopes := []string{"openid"}
if tt.offline {
scopes = append(scopes, "offline_access")
}
for _, client := range tt.crossClients {
scopes = append(scopes, scope.ScopeGoogleCrossClient+client)
}
sessionID, err := sm.NewSession("bogus_idpc", tt.clientID, "bogus", url.URL{}, "", false, scopes)
if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err)
}
_, err = sm.AttachRemoteIdentity(sessionID, oidc.Identity{})
if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err)
}
_, err = sm.AttachUser(sessionID, "ID-1")
if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err)
}
key, err := sm.NewSessionKey(sessionID)
if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err)
}
jwt, token, err := f.srv.CodeToken(f.clientCreds[tt.clientID], key)
if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err)
}
if jwt == nil {
t.Fatalf("case %d: expect non-nil jwt", i)
}
if token != tt.refreshToken {
t.Errorf("case %d: expect refresh token %q, got %q", i, tt.refreshToken, token)
}
claims, err := jwt.Claims()
if err != nil {
t.Fatalf("case %d: unexpected error getting claims: %v", i, err)
}
var gotAUD []string
if len(tt.wantAUD) < 2 {
aud, _, err := claims.StringClaim("aud")
if err != nil {
t.Fatalf("case %d: unexpected error getting 'aud': %q: raw: %v", i, err, claims["aud"])
}
gotAUD = []string{aud}
} else {
gotAUD, _, err = claims.StringsClaim("aud")
if err != nil {
t.Fatalf("case %d: unexpected error getting 'aud': %v", i, err)
}
}
sort.Strings(gotAUD)
if diff := pretty.Compare(tt.wantAUD, gotAUD); diff != "" {
t.Fatalf("case %d: pretty.Compare(tt.wantAUD, gotAUD): %v", i, diff)
}
gotAZP, _, err := claims.StringClaim("azp")
if err != nil {
if err != nil {
t.Fatalf("case %d: unexpected error getting 'aud': %v", i, err)
}
}
if gotAZP != tt.wantAZP {
t.Errorf("case %d: wantAZP=%v, gotAZP=%v", i, tt.wantAZP, gotAZP)
}
}
}

View file

@ -7,7 +7,6 @@ import (
"io" "io"
"net/http" "net/http"
"net/url" "net/url"
"sort"
"strings" "strings"
"time" "time"
@ -22,6 +21,7 @@ import (
"github.com/coreos/dex/connector" "github.com/coreos/dex/connector"
phttp "github.com/coreos/dex/pkg/http" phttp "github.com/coreos/dex/pkg/http"
"github.com/coreos/dex/pkg/log" "github.com/coreos/dex/pkg/log"
"github.com/coreos/dex/scope"
) )
const ( const (
@ -392,20 +392,18 @@ func handleAuthFunc(srv DexServer, idpcs []connector.Connector, tpl *template.Te
func validateScopes(srv DexServer, clientID string, scopes []string) error { func validateScopes(srv DexServer, clientID string, scopes []string) error {
foundOpenIDScope := false foundOpenIDScope := false
sort.Strings(scopes) for i, curScope := range scopes {
for i, scope := range scopes { if i > 0 && curScope == scopes[i-1] {
if i > 0 && scope == scopes[i-1] {
err := oauth2.NewError(oauth2.ErrorInvalidRequest) err := oauth2.NewError(oauth2.ErrorInvalidRequest)
err.Description = fmt.Sprintf( err.Description = fmt.Sprintf(
"Duplicate scopes are not allowed: %q", "Duplicate scopes are not allowed: %q",
scope) curScope)
return err return err
} }
switch { switch {
case strings.HasPrefix(scope, ScopeGoogleCrossClient): case strings.HasPrefix(curScope, scope.ScopeGoogleCrossClient):
otherClient := scope[len(ScopeGoogleCrossClient):] otherClient := curScope[len(scope.ScopeGoogleCrossClient):]
var allowed bool var allowed bool
var err error var err error
if otherClient == clientID { if otherClient == clientID {
@ -424,11 +422,11 @@ func validateScopes(srv DexServer, clientID string, scopes []string) error {
clientID, otherClient) clientID, otherClient)
return err return err
} }
case scope == "openid": case curScope == "openid":
foundOpenIDScope = true foundOpenIDScope = true
case scope == "profile": case curScope == "profile":
case scope == "email": case curScope == "email":
case scope == "offline_access": case curScope == "offline_access":
// According to the spec, for offline_access scope, the client must // According to the spec, for offline_access scope, the client must
// use a response_type value that would result in an Authorization // use a response_type value that would result in an Authorization
// Code. Currently oauth2.ResponseTypeCode is the only supported // Code. Currently oauth2.ResponseTypeCode is the only supported
@ -439,7 +437,7 @@ func validateScopes(srv DexServer, clientID string, scopes []string) error {
default: default:
// Reject all other scopes. // Reject all other scopes.
err := oauth2.NewError(oauth2.ErrorInvalidRequest) err := oauth2.NewError(oauth2.ErrorInvalidRequest)
err.Description = fmt.Sprintf("%q is not a recognized scope", scope) err.Description = fmt.Sprintf("%q is not a recognized scope", curScope)
return err return err
} }
} }

View file

@ -18,6 +18,7 @@ import (
"github.com/coreos/dex/client" "github.com/coreos/dex/client"
"github.com/coreos/dex/connector" "github.com/coreos/dex/connector"
"github.com/coreos/dex/scope"
"github.com/coreos/go-oidc/jose" "github.com/coreos/go-oidc/jose"
"github.com/coreos/go-oidc/oauth2" "github.com/coreos/go-oidc/oauth2"
"github.com/coreos/go-oidc/oidc" "github.com/coreos/go-oidc/oidc"
@ -346,21 +347,21 @@ func TestValidateScopes(t *testing.T) {
{ {
// ERR: invalid cross client auth // ERR: invalid cross client auth
clientID: "XXX", clientID: "XXX",
scopes: []string{"openid", ScopeGoogleCrossClient + "client_a"}, scopes: []string{"openid", scope.ScopeGoogleCrossClient + "client_a"},
wantErr: true, wantErr: true,
}, },
{ {
// OK: valid cross client auth (though perverse - a client // OK: valid cross client auth (though perverse - a client
// requesting cross-client auth for itself) // requesting cross-client auth for itself)
clientID: "client_a", clientID: "client_a",
scopes: []string{"openid", ScopeGoogleCrossClient + "client_a"}, scopes: []string{"openid", scope.ScopeGoogleCrossClient + "client_a"},
wantErr: false, wantErr: false,
}, },
{ {
// OK: valid cross client auth // OK: valid cross client auth
clientID: "client_a", clientID: "client_a",
scopes: []string{"openid", ScopeGoogleCrossClient + "client_b"}, scopes: []string{"openid", scope.ScopeGoogleCrossClient + "client_b"},
wantErr: false, wantErr: false,
}, },
{ {
@ -368,8 +369,8 @@ func TestValidateScopes(t *testing.T) {
// ERR: valid cross client auth...but duplicated scope. // ERR: valid cross client auth...but duplicated scope.
clientID: "client_a", clientID: "client_a",
scopes: []string{"openid", scopes: []string{"openid",
ScopeGoogleCrossClient + "client_b", scope.ScopeGoogleCrossClient + "client_b",
ScopeGoogleCrossClient + "client_b", scope.ScopeGoogleCrossClient + "client_b",
}, },
wantErr: true, wantErr: true,
}, },
@ -378,9 +379,9 @@ func TestValidateScopes(t *testing.T) {
clientID: "client_a", clientID: "client_a",
scopes: []string{ scopes: []string{
"openid", "openid",
ScopeGoogleCrossClient + "client_a", scope.ScopeGoogleCrossClient + "client_a",
ScopeGoogleCrossClient + "client_b", scope.ScopeGoogleCrossClient + "client_b",
ScopeGoogleCrossClient + "client_c", scope.ScopeGoogleCrossClient + "client_c",
}, },
wantErr: false, wantErr: false,
}, },
@ -388,9 +389,9 @@ func TestValidateScopes(t *testing.T) {
// ERR: valid cross client auth with >1 clients including itself...but no openid! // ERR: valid cross client auth with >1 clients including itself...but no openid!
clientID: "client_a", clientID: "client_a",
scopes: []string{ scopes: []string{
ScopeGoogleCrossClient + "client_a", scope.ScopeGoogleCrossClient + "client_a",
ScopeGoogleCrossClient + "client_b", scope.ScopeGoogleCrossClient + "client_b",
ScopeGoogleCrossClient + "client_c", scope.ScopeGoogleCrossClient + "client_c",
}, },
wantErr: true, wantErr: true,
}, },

View file

@ -39,10 +39,6 @@ const (
ResetPasswordTemplateName = "reset-password.html" ResetPasswordTemplateName = "reset-password.html"
APIVersion = "v1" APIVersion = "v1"
// Scope prefix which indicates initiation of a cross-client authentication flow.
// See https://developers.google.com/identity/protocols/CrossClientAuth
ScopeGoogleCrossClient = "audience:server:client_id:"
) )
type OIDCServer interface { type OIDCServer interface {
@ -454,6 +450,36 @@ func (s *Server) CodeToken(creds oidc.ClientCredentials, sessionKey string) (*jo
claims := ses.Claims(s.IssuerURL.String()) claims := ses.Claims(s.IssuerURL.String())
user.AddToClaims(claims) user.AddToClaims(claims)
crossClientIDs := ses.Scope.CrossClientIDs()
if len(crossClientIDs) > 0 {
var aud []string
for _, id := range crossClientIDs {
if ses.ClientID == id {
aud = append(aud, id)
continue
}
allowed, err := s.CrossClientAuthAllowed(ses.ClientID, id)
if err != nil {
log.Errorf("Failed to check cross client auth. reqClientID %v; authClient:ID %v; err: %v", ses.ClientID, id, err)
return nil, "", oauth2.NewError(oauth2.ErrorServerError)
}
if !allowed {
err := oauth2.NewError(oauth2.ErrorInvalidRequest)
err.Description = fmt.Sprintf(
"%q is not authorized to perform cross-client requests for %q",
ses.ClientID, id)
return nil, "", err
}
aud = append(aud, id)
}
if len(aud) == 1 {
claims.Add("aud", aud[0])
} else {
claims.Add("aud", aud)
}
claims.Add("azp", ses.ClientID)
}
jwt, err := jose.NewSignedJWT(claims, signer) jwt, err := jose.NewSignedJWT(claims, signer)
if err != nil { if err != nil {
log.Errorf("Failed to generate ID token: %v", err) log.Errorf("Failed to generate ID token: %v", err)
@ -538,7 +564,7 @@ func (s *Server) RefreshToken(creds oidc.ClientCredentials, token string) (*jose
} }
func (s *Server) CrossClientAuthAllowed(requestingClientID, authorizingClientID string) (bool, error) { func (s *Server) CrossClientAuthAllowed(requestingClientID, authorizingClientID string) (bool, error) {
alloweds, err := s.ClientRepo.GetTrustedPeers(authorizingClientID) alloweds, err := s.ClientRepo.GetTrustedPeers(nil, authorizingClientID)
if err != nil { if err != nil {
return false, err return false, err
} }

View file

@ -9,16 +9,17 @@ import (
"testing" "testing"
"time" "time"
"github.com/coreos/dex/client"
"github.com/coreos/dex/db"
"github.com/coreos/dex/refresh/refreshtest"
"github.com/coreos/dex/session/manager"
"github.com/coreos/dex/user"
"github.com/coreos/go-oidc/jose" "github.com/coreos/go-oidc/jose"
"github.com/coreos/go-oidc/key" "github.com/coreos/go-oidc/key"
"github.com/coreos/go-oidc/oauth2" "github.com/coreos/go-oidc/oauth2"
"github.com/coreos/go-oidc/oidc" "github.com/coreos/go-oidc/oidc"
"github.com/kylelemons/godebug/pretty" "github.com/kylelemons/godebug/pretty"
"github.com/coreos/dex/client"
"github.com/coreos/dex/db"
"github.com/coreos/dex/refresh/refreshtest"
"github.com/coreos/dex/session/manager"
"github.com/coreos/dex/user"
) )
var validRedirURL = url.URL{ var validRedirURL = url.URL{
@ -266,6 +267,12 @@ func TestServerLoginDisabledUser(t *testing.T) {
} }
func TestServerCodeToken(t *testing.T) { func TestServerCodeToken(t *testing.T) {
f, err := makeTestFixtures()
if err != nil {
t.Fatalf("Error creating test fixtures: %v", err)
}
sm := f.sessionManager
tests := []struct { tests := []struct {
scope []string scope []string
refreshToken string refreshToken string
@ -277,21 +284,14 @@ func TestServerCodeToken(t *testing.T) {
}, },
// Have 'offline_access' in scope, should get non-empty refresh token. // Have 'offline_access' in scope, should get non-empty refresh token.
{ {
// NOTE(ericchiang): This test assumes that the database ID of the first // NOTE(ericchiang): This test assumes that the database ID of the
// refresh token will be "1". // first refresh token will be "1".
scope: []string{"openid", "offline_access"}, scope: []string{"openid", "offline_access"},
refreshToken: fmt.Sprintf("1/%s", base64.URLEncoding.EncodeToString([]byte("refresh-1"))), refreshToken: fmt.Sprintf("1/%s", base64.URLEncoding.EncodeToString([]byte("refresh-1"))),
}, },
} }
for i, tt := range tests { for i, tt := range tests {
f, err := makeTestFixtures()
if err != nil {
t.Fatalf("error making test fixtures: %v", err)
}
f.srv.RefreshTokenRepo = refreshtest.NewTestRefreshTokenRepo()
sm := f.sessionManager
sessionID, err := sm.NewSession("bogus_idpc", testClientID, "bogus", url.URL{}, "", false, tt.scope) sessionID, err := sm.NewSession("bogus_idpc", testClientID, "bogus", url.URL{}, "", false, tt.scope)
if err != nil { if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err) t.Fatalf("case %d: unexpected error: %v", i, err)
@ -311,11 +311,9 @@ func TestServerCodeToken(t *testing.T) {
t.Fatalf("case %d: unexpected error: %v", i, err) t.Fatalf("case %d: unexpected error: %v", i, err)
} }
jwt, token, err := f.srv.CodeToken( jwt, token, err := f.srv.CodeToken(oidc.ClientCredentials{
oidc.ClientCredentials{
ID: testClientID, ID: testClientID,
Secret: clientTestSecret, Secret: clientTestSecret}, key)
}, key)
if err != nil { if err != nil {
t.Fatalf("case %d: unexpected error: %v", i, err) t.Fatalf("case %d: unexpected error: %v", i, err)
} }

View file

@ -14,6 +14,7 @@ import (
"github.com/coreos/dex/connector" "github.com/coreos/dex/connector"
"github.com/coreos/dex/db" "github.com/coreos/dex/db"
"github.com/coreos/dex/email" "github.com/coreos/dex/email"
"github.com/coreos/dex/refresh/refreshtest"
sessionmanager "github.com/coreos/dex/session/manager" sessionmanager "github.com/coreos/dex/session/manager"
"github.com/coreos/dex/user" "github.com/coreos/dex/user"
useremail "github.com/coreos/dex/user/email" useremail "github.com/coreos/dex/user/email"
@ -83,6 +84,11 @@ var (
} }
testPrivKey, _ = key.GeneratePrivateKey() testPrivKey, _ = key.GeneratePrivateKey()
testClientCreds = oidc.ClientCredentials{
ID: testClientID,
Secret: base64.URLEncoding.EncodeToString([]byte("secret")),
}
) )
type testFixtures struct { type testFixtures struct {
@ -93,6 +99,7 @@ type testFixtures struct {
redirectURL url.URL redirectURL url.URL
clientRepo client.ClientRepo clientRepo client.ClientRepo
clientManager *clientmanager.ClientManager clientManager *clientmanager.ClientManager
clientCreds map[string]oidc.ClientCredentials
} }
type testFixtureOptions struct { type testFixtureOptions struct {
@ -150,6 +157,8 @@ func makeTestFixturesWithOptions(options testFixtureOptions) (*testFixtures, err
sessionManager := sessionmanager.NewSessionManager(db.NewSessionRepo(db.NewMemDB()), db.NewSessionKeyRepo(db.NewMemDB())) sessionManager := sessionmanager.NewSessionManager(db.NewSessionRepo(db.NewMemDB()), db.NewSessionKeyRepo(db.NewMemDB()))
sessionManager.GenerateCode = sequentialGenerateCodeFunc() sessionManager.GenerateCode = sequentialGenerateCodeFunc()
refreshTokenRepo := refreshtest.NewTestRefreshTokenRepo()
emailer, err := email.NewTemplatizedEmailerFromGlobs( emailer, err := email.NewTemplatizedEmailerFromGlobs(
emailTemplatesLocation+"/*.txt", emailTemplatesLocation+"/*.txt",
emailTemplatesLocation+"/*.html", emailTemplatesLocation+"/*.html",
@ -210,6 +219,7 @@ func makeTestFixturesWithOptions(options testFixtureOptions) (*testFixtures, err
UserManager: userManager, UserManager: userManager,
ClientManager: clientManager, ClientManager: clientManager,
KeyManager: km, KeyManager: km,
RefreshTokenRepo: refreshTokenRepo,
} }
err = setTemplates(srv, tpl) err = setTemplates(srv, tpl)
@ -243,5 +253,8 @@ func makeTestFixturesWithOptions(options testFixtureOptions) (*testFixtures, err
emailer: emailer, emailer: emailer,
clientRepo: clientRepo, clientRepo: clientRepo,
clientManager: clientManager, clientManager: clientManager,
clientCreds: map[string]oidc.ClientCredentials{
testClientID: testClientCreds,
},
}, nil }, nil
} }

View file

@ -6,6 +6,8 @@ import (
"github.com/coreos/go-oidc/jose" "github.com/coreos/go-oidc/jose"
"github.com/coreos/go-oidc/oidc" "github.com/coreos/go-oidc/oidc"
"github.com/coreos/dex/scope"
) )
const ( const (
@ -46,11 +48,13 @@ type Session struct {
// Regsiter indicates that this session is a registration flow. // Regsiter indicates that this session is a registration flow.
Register bool Register bool
// Nonce is optionally provided in the initial authorization request, and propogated in such cases to the generated claims. // Nonce is optionally provided in the initial authorization request, and
// propogated in such cases to the generated claims.
Nonce string Nonce string
// Scope is the 'scope' field in the authentication request. Example scopes are 'openid', 'email', 'offline', etc. // Scope is the 'scope' field in the authentication request. Example scopes
Scope []string // are 'openid', 'email', 'offline', etc.
Scope scope.Scopes
} }
// Claims returns a new set of Claims for the current session. // Claims returns a new set of Claims for the current session.