mirror of
				https://github.com/zitadel/zitadel.git
				synced 2025-10-25 15:29:49 +00:00 
			
		
		
		
	 f3e6f3b23b
			
		
	
	f3e6f3b23b
	
	
	
		
			
			* feat(command): remove org * refactor: imports, unused code, error handling * reduce org removed in action * add org deletion to projections * add org removal to projections * add org removal to projections * org removed projection * lint import * projections * fix: table names in tests * fix: table names in tests * logging * add org state * fix(domain): add Owner removed to object details * feat(ListQuery): add with owner removed * fix(org-delete): add bool to functions to select with owner removed * fix(org-delete): add bools to user grants with events to determine if dependencies lost owner * fix(org-delete): add unit tests for owner removed and org removed events * fix(org-delete): add handling of org remove for grants and members * fix(org-delete): correction of unit tests for owner removed * fix(org-delete): update projections, unit tests and get functions * fix(org-delete): add change date to authnkeys and owner removed to org metadata * fix(org-delete): include owner removed for login names * fix(org-delete): some column fixes in projections and build for queries with owner removed * indexes * fix(org-delete): include review changes * fix(org-delete): change user projection name after merge * fix(org-delete): include review changes for project grant where no project owner is necessary * fix(org-delete): include auth and adminapi tables with owner removed information * fix(org-delete): cleanup username and orgdomain uniqueconstraints when org is removed * fix(org-delete): add permissions for org.remove * remove unnecessary unique constraints * fix column order in primary keys * fix(org-delete): include review changes * fix(org-delete): add owner removed indexes and chang setup step to create tables * fix(org-delete): move PK order of instance_id and change added user_grant from review * fix(org-delete): no params for prepareUserQuery * change to step 6 * merge main * fix(org-delete): OldUserName rename to private * fix linting * cleanup * fix: remove org test * create prerelease * chore: delete org-delete as prerelease Co-authored-by: Stefan Benz <stefan@caos.ch> Co-authored-by: Livio Spring <livio.a@gmail.com> Co-authored-by: Fabi <38692350+hifabienne@users.noreply.github.com> Co-authored-by: Stefan Benz <46600784+stebenz@users.noreply.github.com>
		
			
				
	
	
		
			275 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package oidc
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/zitadel/logging"
 | |
| 	"github.com/zitadel/oidc/v2/pkg/oidc"
 | |
| 	"github.com/zitadel/oidc/v2/pkg/op"
 | |
| 
 | |
| 	"github.com/zitadel/zitadel/internal/api/authz"
 | |
| 	"github.com/zitadel/zitadel/internal/api/http/middleware"
 | |
| 	"github.com/zitadel/zitadel/internal/errors"
 | |
| 	"github.com/zitadel/zitadel/internal/query"
 | |
| 	"github.com/zitadel/zitadel/internal/telemetry/tracing"
 | |
| 	"github.com/zitadel/zitadel/internal/user/model"
 | |
| )
 | |
| 
 | |
| 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) }()
 | |
| 	userAgentID, ok := middleware.UserAgentIDFromCtx(ctx)
 | |
| 	if !ok {
 | |
| 		return nil, errors.ThrowPreconditionFailed(nil, "OIDC-sd436", "no user agent id")
 | |
| 	}
 | |
| 	req.Scopes, err = o.assertProjectRoleScopes(ctx, req.ClientID, req.Scopes)
 | |
| 	if err != nil {
 | |
| 		return nil, errors.ThrowPreconditionFailed(err, "OIDC-Gqrfg", "Errors.Internal")
 | |
| 	}
 | |
| 	authRequest := CreateAuthRequestToBusiness(ctx, req, userAgentID, userID)
 | |
| 	resp, err := o.repo.CreateAuthRequest(ctx, authRequest)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return AuthRequestFromBusiness(resp)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) AuthRequestByID(ctx context.Context, id string) (_ op.AuthRequest, err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 	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)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return AuthRequestFromBusiness(resp)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) AuthRequestByCode(ctx context.Context, code string) (_ op.AuthRequest, err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 
 | |
| 	resp, err := o.repo.AuthRequestByCode(ctx, code)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return AuthRequestFromBusiness(resp)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) SaveAuthCode(ctx context.Context, id, code string) (err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 	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)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) DeleteAuthRequest(ctx context.Context, id string) (err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 
 | |
| 	return o.repo.DeleteAuthRequest(ctx, id)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) CreateAccessToken(ctx context.Context, req op.TokenRequest) (_ string, _ time.Time, err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 	var userAgentID, applicationID, userOrgID string
 | |
| 	authReq, ok := req.(*AuthRequest)
 | |
| 	if ok {
 | |
| 		userAgentID = authReq.AgentID
 | |
| 		applicationID = authReq.ApplicationID
 | |
| 		userOrgID = authReq.UserOrgID
 | |
| 	}
 | |
| 
 | |
| 	accessTokenLifetime, _, _, _, err := o.getOIDCSettings(ctx)
 | |
| 	if err != nil {
 | |
| 		return "", time.Time{}, err
 | |
| 	}
 | |
| 
 | |
| 	resp, err := o.command.AddUserToken(setContextUserSystem(ctx), userOrgID, userAgentID, applicationID, req.GetSubject(), req.GetAudience(), req.GetScopes(), accessTokenLifetime) //PLANNED: lifetime from client
 | |
| 	if err != nil {
 | |
| 		return "", time.Time{}, err
 | |
| 	}
 | |
| 	return resp.TokenID, resp.Expiration, nil
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) CreateAccessAndRefreshTokens(ctx context.Context, req op.TokenRequest, refreshToken string) (_, _ string, _ time.Time, err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 	userAgentID, applicationID, userOrgID, authTime, authMethodsReferences := getInfoFromRequest(req)
 | |
| 	scopes, err := o.assertProjectRoleScopes(ctx, applicationID, req.GetScopes())
 | |
| 	if err != nil {
 | |
| 		return "", "", time.Time{}, errors.ThrowPreconditionFailed(err, "OIDC-Df2fq", "Errors.Internal")
 | |
| 	}
 | |
| 	if request, ok := req.(op.RefreshTokenRequest); ok {
 | |
| 		request.SetCurrentScopes(scopes)
 | |
| 	}
 | |
| 
 | |
| 	accessTokenLifetime, _, refreshTokenIdleExpiration, refreshTokenExpiration, err := o.getOIDCSettings(ctx)
 | |
| 	if err != nil {
 | |
| 		return "", "", time.Time{}, err
 | |
| 	}
 | |
| 
 | |
| 	resp, token, err := o.command.AddAccessAndRefreshToken(setContextUserSystem(ctx), userOrgID, userAgentID, applicationID, req.GetSubject(),
 | |
| 		refreshToken, req.GetAudience(), scopes, authMethodsReferences, accessTokenLifetime,
 | |
| 		refreshTokenIdleExpiration, refreshTokenExpiration, authTime) //PLANNED: lifetime from client
 | |
| 	if err != nil {
 | |
| 		if errors.IsErrorInvalidArgument(err) {
 | |
| 			err = oidc.ErrInvalidGrant().WithParent(err)
 | |
| 		}
 | |
| 		return "", "", time.Time{}, err
 | |
| 	}
 | |
| 	return resp.TokenID, token, resp.Expiration, nil
 | |
| }
 | |
| 
 | |
| func getInfoFromRequest(req op.TokenRequest) (string, string, string, time.Time, []string) {
 | |
| 	authReq, ok := req.(*AuthRequest)
 | |
| 	if ok {
 | |
| 		return authReq.AgentID, authReq.ApplicationID, authReq.UserOrgID, authReq.AuthTime, authReq.GetAMR()
 | |
| 	}
 | |
| 	refreshReq, ok := req.(*RefreshTokenRequest)
 | |
| 	if ok {
 | |
| 		return refreshReq.UserAgentID, refreshReq.ClientID, "", refreshReq.AuthTime, refreshReq.AuthMethodsReferences
 | |
| 	}
 | |
| 	return "", "", "", time.Time{}, nil
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) TokenRequestByRefreshToken(ctx context.Context, refreshToken string) (op.RefreshTokenRequest, error) {
 | |
| 	tokenView, err := o.repo.RefreshTokenByID(ctx, refreshToken)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return RefreshTokenRequestFromBusiness(tokenView), nil
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) TerminateSession(ctx context.Context, userID, clientID string) (err error) {
 | |
| 	ctx, span := tracing.NewSpan(ctx)
 | |
| 	defer func() { span.EndWithError(err) }()
 | |
| 	userAgentID, ok := middleware.UserAgentIDFromCtx(ctx)
 | |
| 	if !ok {
 | |
| 		logging.Error("no user agent id")
 | |
| 		return errors.ThrowPreconditionFailed(nil, "OIDC-fso7F", "no user agent id")
 | |
| 	}
 | |
| 	userIDs, err := o.repo.UserSessionUserIDsByAgentID(ctx, userAgentID)
 | |
| 	if err != nil {
 | |
| 		logging.WithError(err).Error("error retrieving user sessions")
 | |
| 		return err
 | |
| 	}
 | |
| 	if len(userIDs) == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 	data := authz.CtxData{
 | |
| 		UserID: userID,
 | |
| 	}
 | |
| 	err = o.command.HumansSignOut(authz.SetCtxData(ctx, data), userAgentID, userIDs)
 | |
| 	logging.OnError(err).Error("error signing out")
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) RevokeToken(ctx context.Context, token, userID, clientID string) *oidc.Error {
 | |
| 	refreshToken, err := o.repo.RefreshTokenByID(ctx, token)
 | |
| 	if err == nil {
 | |
| 		if refreshToken.ClientID != clientID {
 | |
| 			return oidc.ErrInvalidClient().WithDescription("token was not issued for this client")
 | |
| 		}
 | |
| 		_, err = o.command.RevokeRefreshToken(ctx, refreshToken.UserID, refreshToken.ResourceOwner, refreshToken.ID)
 | |
| 		if err == nil || errors.IsNotFound(err) {
 | |
| 			return nil
 | |
| 		}
 | |
| 		return oidc.ErrServerError().WithParent(err)
 | |
| 	}
 | |
| 	accessToken, err := o.repo.TokenByIDs(ctx, userID, token)
 | |
| 	if err != nil {
 | |
| 		if errors.IsNotFound(err) {
 | |
| 			return nil
 | |
| 		}
 | |
| 		return oidc.ErrServerError().WithParent(err)
 | |
| 	}
 | |
| 	if accessToken.ApplicationID != clientID {
 | |
| 		return oidc.ErrInvalidClient().WithDescription("token was not issued for this client")
 | |
| 	}
 | |
| 	_, err = o.command.RevokeAccessToken(ctx, userID, accessToken.ResourceOwner, accessToken.ID)
 | |
| 	if err == nil || errors.IsNotFound(err) {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return oidc.ErrServerError().WithParent(err)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) assertProjectRoleScopes(ctx context.Context, clientID string, scopes []string) ([]string, error) {
 | |
| 	for _, scope := range scopes {
 | |
| 		if strings.HasPrefix(scope, ScopeProjectRolePrefix) {
 | |
| 			return scopes, nil
 | |
| 		}
 | |
| 	}
 | |
| 	projectID, err := o.query.ProjectIDFromOIDCClientID(ctx, clientID, false)
 | |
| 	if err != nil {
 | |
| 		return nil, errors.ThrowPreconditionFailed(nil, "OIDC-AEG4d", "Errors.Internal")
 | |
| 	}
 | |
| 	project, err := o.query.ProjectByID(ctx, false, projectID, false)
 | |
| 	if err != nil {
 | |
| 		return nil, errors.ThrowPreconditionFailed(nil, "OIDC-w4wIn", "Errors.Internal")
 | |
| 	}
 | |
| 	if !project.ProjectRoleAssertion {
 | |
| 		return scopes, nil
 | |
| 	}
 | |
| 	projectIDQuery, err := query.NewProjectRoleProjectIDSearchQuery(project.ID)
 | |
| 	if err != nil {
 | |
| 		return nil, errors.ThrowInternal(err, "OIDC-Cyc78", "Errors.Internal")
 | |
| 	}
 | |
| 	roles, err := o.query.SearchProjectRoles(ctx, true, &query.ProjectRoleSearchQueries{Queries: []query.SearchQuery{projectIDQuery}}, false)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	for _, role := range roles.ProjectRoles {
 | |
| 		scopes = append(scopes, ScopeProjectRolePrefix+role.Key)
 | |
| 	}
 | |
| 	return scopes, nil
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) assertClientScopesForPAT(ctx context.Context, token *model.TokenView, clientID string) error {
 | |
| 	token.Audience = append(token.Audience, clientID)
 | |
| 	projectID, err := o.query.ProjectIDFromClientID(ctx, clientID, false)
 | |
| 	if err != nil {
 | |
| 		return errors.ThrowPreconditionFailed(nil, "OIDC-AEG4d", "Errors.Internal")
 | |
| 	}
 | |
| 	projectIDQuery, err := query.NewProjectRoleProjectIDSearchQuery(projectID)
 | |
| 	if err != nil {
 | |
| 		return errors.ThrowInternal(err, "OIDC-Cyc78", "Errors.Internal")
 | |
| 	}
 | |
| 	roles, err := o.query.SearchProjectRoles(ctx, true, &query.ProjectRoleSearchQueries{Queries: []query.SearchQuery{projectIDQuery}}, false)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for _, role := range roles.ProjectRoles {
 | |
| 		token.Scopes = append(token.Scopes, ScopeProjectRolePrefix+role.Key)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func setContextUserSystem(ctx context.Context) context.Context {
 | |
| 	data := authz.CtxData{
 | |
| 		UserID: "SYSTEM",
 | |
| 	}
 | |
| 	return authz.SetCtxData(ctx, data)
 | |
| }
 | |
| 
 | |
| func (o *OPStorage) getOIDCSettings(ctx context.Context) (accessTokenLifetime, idTokenLifetime, refreshTokenIdleExpiration, refreshTokenExpiration time.Duration, _ error) {
 | |
| 	oidcSettings, err := o.query.OIDCSettingsByAggID(ctx, authz.GetInstance(ctx).InstanceID())
 | |
| 	if err != nil && !errors.IsNotFound(err) {
 | |
| 		return time.Duration(0), time.Duration(0), time.Duration(0), time.Duration(0), err
 | |
| 	}
 | |
| 
 | |
| 	if oidcSettings != nil {
 | |
| 		return oidcSettings.AccessTokenLifetime, oidcSettings.IdTokenLifetime, oidcSettings.RefreshTokenIdleExpiration, oidcSettings.RefreshTokenExpiration, nil
 | |
| 	}
 | |
| 	return o.defaultAccessTokenLifetime, o.defaultIdTokenLifetime, o.defaultRefreshTokenIdleExpiration, o.defaultRefreshTokenExpiration, nil
 | |
| }
 |