2020-06-05 05:50:04 +00:00
|
|
|
package oidc
|
|
|
|
|
|
|
|
import (
|
2020-11-03 09:50:03 +00:00
|
|
|
authreq_model "github.com/caos/zitadel/internal/auth_request/model"
|
|
|
|
"strings"
|
2020-06-05 05:50:04 +00:00
|
|
|
"time"
|
|
|
|
|
2020-10-16 05:49:38 +00:00
|
|
|
"github.com/caos/oidc/pkg/oidc"
|
2020-06-05 05:50:04 +00:00
|
|
|
"github.com/caos/oidc/pkg/op"
|
|
|
|
|
|
|
|
"github.com/caos/zitadel/internal/errors"
|
|
|
|
"github.com/caos/zitadel/internal/project/model"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Client struct {
|
|
|
|
*model.ApplicationView
|
|
|
|
defaultLoginURL string
|
|
|
|
defaultAccessTokenLifetime time.Duration
|
|
|
|
defaultIdTokenLifetime time.Duration
|
2020-10-16 05:49:38 +00:00
|
|
|
allowedScopes []string
|
2020-06-05 05:50:04 +00:00
|
|
|
}
|
|
|
|
|
2020-10-16 05:49:38 +00:00
|
|
|
func ClientFromBusiness(app *model.ApplicationView, defaultLoginURL string, defaultAccessTokenLifetime, defaultIdTokenLifetime time.Duration, allowedScopes []string) (op.Client, error) {
|
2020-06-05 05:50:04 +00:00
|
|
|
if !app.IsOIDC {
|
|
|
|
return nil, errors.ThrowInvalidArgument(nil, "OIDC-d5bhD", "client is not a proper oidc application")
|
|
|
|
}
|
2020-10-16 05:49:38 +00:00
|
|
|
return &Client{
|
|
|
|
ApplicationView: app,
|
|
|
|
defaultLoginURL: defaultLoginURL,
|
|
|
|
defaultAccessTokenLifetime: defaultAccessTokenLifetime,
|
|
|
|
defaultIdTokenLifetime: defaultIdTokenLifetime,
|
|
|
|
allowedScopes: allowedScopes},
|
|
|
|
nil
|
2020-06-05 05:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) ApplicationType() op.ApplicationType {
|
|
|
|
return op.ApplicationType(c.OIDCApplicationType)
|
|
|
|
}
|
|
|
|
|
2020-08-10 07:34:56 +00:00
|
|
|
func (c *Client) AuthMethod() op.AuthMethod {
|
2020-06-05 05:50:04 +00:00
|
|
|
return authMethodToOIDC(c.OIDCAuthMethodType)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) GetID() string {
|
|
|
|
return c.OIDCClientID
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) LoginURL(id string) string {
|
|
|
|
return c.defaultLoginURL + id
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) RedirectURIs() []string {
|
|
|
|
return c.OIDCRedirectUris
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) PostLogoutRedirectURIs() []string {
|
|
|
|
return c.OIDCPostLogoutRedirectUris
|
|
|
|
}
|
|
|
|
|
2020-08-10 07:34:56 +00:00
|
|
|
func (c *Client) ResponseTypes() []oidc.ResponseType {
|
|
|
|
return responseTypesToOIDC(c.OIDCResponseTypes)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) DevMode() bool {
|
|
|
|
return c.ApplicationView.DevMode
|
|
|
|
}
|
|
|
|
|
2020-11-03 09:50:03 +00:00
|
|
|
func (c *Client) RestrictAdditionalIdTokenScopes() func(scopes []string) []string {
|
|
|
|
return func(scopes []string) []string {
|
|
|
|
if c.IDTokenRoleAssertion {
|
|
|
|
return scopes
|
|
|
|
}
|
|
|
|
return removeScopeWithPrefix(scopes, ScopeProjectRolePrefix)
|
|
|
|
}
|
2020-10-16 05:49:38 +00:00
|
|
|
}
|
|
|
|
|
2020-11-03 09:50:03 +00:00
|
|
|
func (c *Client) RestrictAdditionalAccessTokenScopes() func(scopes []string) []string {
|
|
|
|
return func(scopes []string) []string {
|
|
|
|
if c.AccessTokenRoleAssertion {
|
|
|
|
return scopes
|
|
|
|
}
|
|
|
|
return removeScopeWithPrefix(scopes, ScopeProjectRolePrefix)
|
|
|
|
}
|
2020-10-16 05:49:38 +00:00
|
|
|
}
|
|
|
|
|
2020-06-05 05:50:04 +00:00
|
|
|
func (c *Client) AccessTokenLifetime() time.Duration {
|
|
|
|
return c.defaultAccessTokenLifetime //PLANNED: impl from real client
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) IDTokenLifetime() time.Duration {
|
|
|
|
return c.defaultIdTokenLifetime //PLANNED: impl from real client
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) AccessTokenType() op.AccessTokenType {
|
2020-10-16 05:49:38 +00:00
|
|
|
return accessTokenTypeToOIDC(c.ApplicationView.AccessTokenType)
|
|
|
|
}
|
|
|
|
|
2020-11-03 09:50:03 +00:00
|
|
|
func (c *Client) IsScopeAllowed(scope string) bool {
|
|
|
|
if strings.HasPrefix(scope, authreq_model.OrgDomainPrimaryScope) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
for _, allowedScope := range c.allowedScopes {
|
|
|
|
if scope == allowedScope {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-10-16 05:49:38 +00:00
|
|
|
func accessTokenTypeToOIDC(tokenType model.OIDCTokenType) op.AccessTokenType {
|
|
|
|
switch tokenType {
|
|
|
|
case model.OIDCTokenTypeBearer:
|
|
|
|
return op.AccessTokenTypeBearer
|
|
|
|
case model.OIDCTokenTypeJWT:
|
|
|
|
return op.AccessTokenTypeJWT
|
|
|
|
default:
|
|
|
|
return op.AccessTokenTypeBearer
|
|
|
|
}
|
2020-06-05 05:50:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func authMethodToOIDC(authType model.OIDCAuthMethodType) op.AuthMethod {
|
|
|
|
switch authType {
|
2020-06-23 12:47:47 +00:00
|
|
|
case model.OIDCAuthMethodTypeBasic:
|
2020-06-05 05:50:04 +00:00
|
|
|
return op.AuthMethodBasic
|
2020-06-23 12:47:47 +00:00
|
|
|
case model.OIDCAuthMethodTypePost:
|
2020-06-05 05:50:04 +00:00
|
|
|
return op.AuthMethodPost
|
2020-06-23 12:47:47 +00:00
|
|
|
case model.OIDCAuthMethodTypeNone:
|
2020-06-05 05:50:04 +00:00
|
|
|
return op.AuthMethodNone
|
|
|
|
default:
|
|
|
|
return op.AuthMethodBasic
|
|
|
|
}
|
|
|
|
}
|
2020-08-10 07:34:56 +00:00
|
|
|
|
|
|
|
func responseTypesToOIDC(responseTypes []model.OIDCResponseType) []oidc.ResponseType {
|
|
|
|
oidcTypes := make([]oidc.ResponseType, len(responseTypes))
|
|
|
|
for i, t := range responseTypes {
|
|
|
|
oidcTypes[i] = responseTypeToOIDC(t)
|
|
|
|
}
|
|
|
|
return oidcTypes
|
|
|
|
}
|
|
|
|
|
|
|
|
func responseTypeToOIDC(responseType model.OIDCResponseType) oidc.ResponseType {
|
|
|
|
switch responseType {
|
|
|
|
case model.OIDCResponseTypeCode:
|
|
|
|
return oidc.ResponseTypeCode
|
|
|
|
case model.OIDCResponseTypeIDTokenToken:
|
|
|
|
return oidc.ResponseTypeIDToken
|
|
|
|
case model.OIDCResponseTypeIDToken:
|
|
|
|
return oidc.ResponseTypeIDTokenOnly
|
|
|
|
default:
|
|
|
|
return oidc.ResponseTypeCode
|
|
|
|
}
|
|
|
|
}
|
2020-11-03 09:50:03 +00:00
|
|
|
|
|
|
|
func removeScopeWithPrefix(scopes []string, scopePrefix ...string) []string {
|
|
|
|
newScopeList := make([]string, 0)
|
|
|
|
for _, scope := range scopes {
|
|
|
|
hasPrefix := false
|
|
|
|
for _, prefix := range scopePrefix {
|
|
|
|
if strings.HasPrefix(scope, prefix) {
|
|
|
|
hasPrefix = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !hasPrefix {
|
|
|
|
newScopeList = append(newScopeList, scope)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return newScopeList
|
|
|
|
}
|