2020-06-05 07:50:04 +02:00
package oidc
import (
"context"
2020-07-09 14:05:12 +02:00
"fmt"
2020-10-16 07:49:38 +02:00
"strings"
2020-06-05 07:50:04 +02:00
"time"
"github.com/caos/oidc/pkg/oidc"
"github.com/caos/oidc/pkg/op"
"gopkg.in/square/go-jose.v2"
2020-08-31 08:49:35 +02:00
"github.com/caos/zitadel/internal/api/http/middleware"
2020-06-05 07:50:04 +02:00
"github.com/caos/zitadel/internal/errors"
2020-10-16 07:49:38 +02:00
proj_model "github.com/caos/zitadel/internal/project/model"
2020-12-02 08:50:59 +01:00
"github.com/caos/zitadel/internal/telemetry/tracing"
2020-07-09 14:05:12 +02:00
grant_model "github.com/caos/zitadel/internal/usergrant/model"
2020-06-05 07:50:04 +02:00
)
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) CreateAuthRequest ( ctx context . Context , req * oidc . AuthRequest , userID string ) ( _ op . AuthRequest , err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-08-31 08:49:35 +02:00
userAgentID , ok := middleware . UserAgentIDFromCtx ( ctx )
2020-06-05 07:50:04 +02:00
if ! ok {
return nil , errors . ThrowPreconditionFailed ( nil , "OIDC-sd436" , "no user agent id" )
}
2020-10-16 07:49:38 +02:00
app , err := o . repo . ApplicationByClientID ( ctx , req . ClientID )
if err != nil {
return nil , errors . ThrowPreconditionFailed ( nil , "OIDC-AEG4d" , "Errors.Internal" )
}
req . Scopes , err = o . assertProjectRoleScopes ( app , req . Scopes )
if err != nil {
return nil , errors . ThrowPreconditionFailed ( nil , "OIDC-Gqrfg" , "Errors.Internal" )
}
2020-06-05 07:50:04 +02:00
authRequest := CreateAuthRequestToBusiness ( ctx , req , userAgentID , userID )
resp , err := o . repo . CreateAuthRequest ( ctx , authRequest )
if err != nil {
return nil , err
}
return AuthRequestFromBusiness ( resp )
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) AuthRequestByID ( ctx context . Context , id string ) ( _ op . AuthRequest , err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-08-31 08:49:35 +02:00
userAgentID , ok := middleware . UserAgentIDFromCtx ( ctx )
if ! ok {
return nil , errors . ThrowPreconditionFailed ( nil , "OIDC-D3g21" , "no user agent id" )
}
resp , err := o . repo . AuthRequestByIDCheckLoggedIn ( ctx , id , userAgentID )
2020-06-05 07:50:04 +02:00
if err != nil {
return nil , err
}
return AuthRequestFromBusiness ( resp )
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) AuthRequestByCode ( ctx context . Context , code string ) ( _ op . AuthRequest , err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-06-05 07:50:04 +02:00
resp , err := o . repo . AuthRequestByCode ( ctx , code )
if err != nil {
return nil , err
}
return AuthRequestFromBusiness ( resp )
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) SaveAuthCode ( ctx context . Context , id , code string ) ( err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-08-31 08:49:35 +02:00
userAgentID , ok := middleware . UserAgentIDFromCtx ( ctx )
if ! ok {
return errors . ThrowPreconditionFailed ( nil , "OIDC-Dgus2" , "no user agent id" )
}
return o . repo . SaveAuthCode ( ctx , id , code , userAgentID )
2020-06-05 07:50:04 +02:00
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) DeleteAuthRequest ( ctx context . Context , id string ) ( err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-06-05 07:50:04 +02:00
return o . repo . DeleteAuthRequest ( ctx , id )
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) CreateToken ( ctx context . Context , req op . TokenRequest ) ( _ string , _ time . Time , err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-09-17 08:49:33 +02:00
var userAgentID , applicationID string
authReq , ok := req . ( * AuthRequest )
if ok {
userAgentID = authReq . AgentID
applicationID = authReq . ApplicationID
2020-07-09 14:05:12 +02:00
}
2020-09-17 08:49:33 +02:00
resp , err := o . repo . CreateToken ( ctx , userAgentID , applicationID , req . GetSubject ( ) , req . GetAudience ( ) , req . GetScopes ( ) , o . defaultAccessTokenLifetime ) //PLANNED: lifetime from client
2020-06-05 07:50:04 +02:00
if err != nil {
return "" , time . Time { } , err
}
2020-10-15 13:52:41 +02:00
return resp . TokenID , resp . Expiration , nil
2020-06-05 07:50:04 +02:00
}
2020-07-09 14:05:12 +02:00
func grantsToScopes ( grants [ ] * grant_model . UserGrantView ) [ ] string {
scopes := make ( [ ] string , 0 )
for _ , grant := range grants {
for _ , role := range grant . RoleKeys {
scopes = append ( scopes , fmt . Sprintf ( "%v:%v" , grant . ResourceOwner , role ) )
}
}
return scopes
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) TerminateSession ( ctx context . Context , userID , clientID string ) ( err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-08-31 08:49:35 +02:00
userAgentID , ok := middleware . UserAgentIDFromCtx ( ctx )
2020-06-05 07:50:04 +02:00
if ! ok {
return errors . ThrowPreconditionFailed ( nil , "OIDC-fso7F" , "no user agent id" )
}
2020-06-29 09:49:40 +02:00
return o . repo . SignOut ( ctx , userAgentID )
2020-06-05 07:50:04 +02:00
}
func ( o * OPStorage ) GetSigningKey ( ctx context . Context , keyCh chan <- jose . SigningKey , errCh chan <- error , timer <- chan time . Time ) {
o . repo . GetSigningKey ( ctx , keyCh , errCh , timer )
}
2020-10-21 10:18:34 +02:00
func ( o * OPStorage ) GetKeySet ( ctx context . Context ) ( _ * jose . JSONWebKeySet , err error ) {
ctx , span := tracing . NewSpan ( ctx )
defer func ( ) { span . EndWithError ( err ) } ( )
2020-06-05 07:50:04 +02:00
return o . repo . GetKeySet ( ctx )
}
func ( o * OPStorage ) SaveNewKeyPair ( ctx context . Context ) error {
return o . repo . GenerateSigningKeyPair ( ctx , o . signingKeyAlgorithm )
}
2020-10-16 07:49:38 +02:00
func ( o * OPStorage ) assertProjectRoleScopes ( app * proj_model . ApplicationView , scopes [ ] string ) ( [ ] string , error ) {
if ! app . ProjectRoleAssertion {
return scopes , nil
}
for _ , scope := range scopes {
if strings . HasPrefix ( scope , ScopeProjectRolePrefix ) {
return scopes , nil
}
}
roles , err := o . repo . ProjectRolesByProjectID ( app . ProjectID )
if err != nil {
return nil , err
}
for _ , role := range roles {
scopes = append ( scopes , ScopeProjectRolePrefix + role . Key )
}
return scopes , nil
}