mirror of
https://github.com/zitadel/zitadel.git
synced 2025-07-13 11:18:34 +00:00

# Which Problems Are Solved Currently if a user signs in using an IdP, once they sign out of Zitadel, the corresponding IdP session is not terminated. This can be the desired behavior. In some cases, e.g. when using a shared computer it results in a potential security risk, since a follower user might be able to sign in as the previous using the still open IdP session. # How the Problems Are Solved - Admins can enabled a federated logout option on SAML IdPs through the Admin and Management APIs. - During the termination of a login V1 session using OIDC end_session endpoint, Zitadel will check if an IdP was used to authenticate that session. - In case there was a SAML IdP used with Federated Logout enabled, it will intercept the logout process, store the information into the shared cache and redirect to the federated logout endpoint in the V1 login. - The V1 login federated logout endpoint checks every request on an existing cache entry. On success it will create a SAML logout request for the used IdP and either redirect or POST to the configured SLO endpoint. The cache entry is updated with a `redirected` state. - A SLO endpoint is added to the `/idp` handlers, which will handle the SAML logout responses. At the moment it will check again for an existing federated logout entry (with state `redirected`) in the cache. On success, the user is redirected to the initially provided `post_logout_redirect_uri` from the end_session request. # Additional Changes None # Additional Context - This PR merges the https://github.com/zitadel/zitadel/pull/9841 and https://github.com/zitadel/zitadel/pull/9854 to main, additionally updating the docs on Entra ID SAML. - closes #9228 - backport to 3.x --------- Co-authored-by: Silvan <27845747+adlerhurst@users.noreply.github.com> Co-authored-by: Zach Hirschtritt <zachary.hirschtritt@klaviyo.com>
528 lines
18 KiB
Go
528 lines
18 KiB
Go
package admin
|
|
|
|
import (
|
|
"github.com/crewjam/saml"
|
|
"github.com/muhlemmer/gu"
|
|
|
|
idp_grpc "github.com/zitadel/zitadel/internal/api/grpc/idp"
|
|
"github.com/zitadel/zitadel/internal/api/grpc/object"
|
|
"github.com/zitadel/zitadel/internal/command"
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
"github.com/zitadel/zitadel/internal/eventstore/v1/models"
|
|
"github.com/zitadel/zitadel/internal/query"
|
|
"github.com/zitadel/zitadel/internal/zerrors"
|
|
admin_pb "github.com/zitadel/zitadel/pkg/grpc/admin"
|
|
idp_pb "github.com/zitadel/zitadel/pkg/grpc/idp"
|
|
)
|
|
|
|
func addOIDCIDPRequestToDomain(req *admin_pb.AddOIDCIDPRequest) *domain.IDPConfig {
|
|
return &domain.IDPConfig{
|
|
Name: req.Name,
|
|
OIDCConfig: addOIDCIDPRequestToDomainOIDCIDPConfig(req),
|
|
StylingType: idp_grpc.IDPStylingTypeToDomain(req.StylingType),
|
|
Type: domain.IDPConfigTypeOIDC,
|
|
AutoRegister: req.AutoRegister,
|
|
}
|
|
}
|
|
|
|
func addOIDCIDPRequestToDomainOIDCIDPConfig(req *admin_pb.AddOIDCIDPRequest) *domain.OIDCIDPConfig {
|
|
return &domain.OIDCIDPConfig{
|
|
ClientID: req.ClientId,
|
|
ClientSecretString: req.ClientSecret,
|
|
Issuer: req.Issuer,
|
|
Scopes: req.Scopes,
|
|
IDPDisplayNameMapping: idp_grpc.MappingFieldToDomain(req.DisplayNameMapping),
|
|
UsernameMapping: idp_grpc.MappingFieldToDomain(req.UsernameMapping),
|
|
}
|
|
}
|
|
|
|
func addJWTIDPRequestToDomain(req *admin_pb.AddJWTIDPRequest) *domain.IDPConfig {
|
|
return &domain.IDPConfig{
|
|
Name: req.Name,
|
|
JWTConfig: addJWTIDPRequestToDomainJWTIDPConfig(req),
|
|
StylingType: idp_grpc.IDPStylingTypeToDomain(req.StylingType),
|
|
Type: domain.IDPConfigTypeJWT,
|
|
AutoRegister: req.AutoRegister,
|
|
}
|
|
}
|
|
|
|
func addJWTIDPRequestToDomainJWTIDPConfig(req *admin_pb.AddJWTIDPRequest) *domain.JWTIDPConfig {
|
|
return &domain.JWTIDPConfig{
|
|
JWTEndpoint: req.JwtEndpoint,
|
|
Issuer: req.Issuer,
|
|
KeysEndpoint: req.KeysEndpoint,
|
|
HeaderName: req.HeaderName,
|
|
}
|
|
}
|
|
|
|
func updateIDPToDomain(req *admin_pb.UpdateIDPRequest) *domain.IDPConfig {
|
|
return &domain.IDPConfig{
|
|
IDPConfigID: req.IdpId,
|
|
Name: req.Name,
|
|
StylingType: idp_grpc.IDPStylingTypeToDomain(req.StylingType),
|
|
AutoRegister: req.AutoRegister,
|
|
}
|
|
}
|
|
|
|
func updateOIDCConfigToDomain(req *admin_pb.UpdateIDPOIDCConfigRequest) *domain.OIDCIDPConfig {
|
|
return &domain.OIDCIDPConfig{
|
|
IDPConfigID: req.IdpId,
|
|
ClientID: req.ClientId,
|
|
ClientSecretString: req.ClientSecret,
|
|
Issuer: req.Issuer,
|
|
Scopes: req.Scopes,
|
|
IDPDisplayNameMapping: idp_grpc.MappingFieldToDomain(req.DisplayNameMapping),
|
|
UsernameMapping: idp_grpc.MappingFieldToDomain(req.UsernameMapping),
|
|
}
|
|
}
|
|
|
|
func updateJWTConfigToDomain(req *admin_pb.UpdateIDPJWTConfigRequest) *domain.JWTIDPConfig {
|
|
return &domain.JWTIDPConfig{
|
|
IDPConfigID: req.IdpId,
|
|
JWTEndpoint: req.JwtEndpoint,
|
|
Issuer: req.Issuer,
|
|
KeysEndpoint: req.KeysEndpoint,
|
|
HeaderName: req.HeaderName,
|
|
}
|
|
}
|
|
|
|
func listIDPsToModel(instanceID string, req *admin_pb.ListIDPsRequest) (*query.IDPSearchQueries, error) {
|
|
offset, limit, asc := object.ListQueryToModel(req.Query)
|
|
queries, err := idpQueriesToModel(req.Queries)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
iamQuery, err := query.NewIDPResourceOwnerSearchQuery(instanceID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
queries = append(queries, iamQuery)
|
|
return &query.IDPSearchQueries{
|
|
SearchRequest: query.SearchRequest{
|
|
Offset: offset,
|
|
Limit: limit,
|
|
Asc: asc,
|
|
SortingColumn: idp_grpc.FieldNameToModel(req.SortingColumn),
|
|
},
|
|
Queries: queries,
|
|
}, nil
|
|
}
|
|
|
|
func idpQueriesToModel(queries []*admin_pb.IDPQuery) (q []query.SearchQuery, err error) {
|
|
q = make([]query.SearchQuery, len(queries))
|
|
for i, query := range queries {
|
|
q[i], err = idpQueryToModel(query)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return q, nil
|
|
}
|
|
|
|
func idpQueryToModel(idpQuery *admin_pb.IDPQuery) (query.SearchQuery, error) {
|
|
switch q := idpQuery.Query.(type) {
|
|
case *admin_pb.IDPQuery_IdpNameQuery:
|
|
return query.NewIDPNameSearchQuery(object.TextMethodToQuery(q.IdpNameQuery.Method), q.IdpNameQuery.Name)
|
|
case *admin_pb.IDPQuery_IdpIdQuery:
|
|
return query.NewIDPIDSearchQuery(q.IdpIdQuery.Id)
|
|
default:
|
|
return nil, zerrors.ThrowInvalidArgument(nil, "ADMIN-VmqQu", "List.Query.Invalid")
|
|
}
|
|
}
|
|
|
|
func idpsToDomain(idps []*query.IDP) []*domain.IDPProvider {
|
|
idpProvider := make([]*domain.IDPProvider, len(idps))
|
|
for i, idp := range idps {
|
|
idpProvider[i] = &domain.IDPProvider{
|
|
ObjectRoot: models.ObjectRoot{
|
|
AggregateID: idp.ResourceOwner,
|
|
},
|
|
IDPConfigID: idp.ID,
|
|
Type: idp.OwnerType,
|
|
}
|
|
}
|
|
return idpProvider
|
|
}
|
|
|
|
func idpUserLinksToDomain(idps []*query.IDPUserLink) []*domain.UserIDPLink {
|
|
externalIDPs := make([]*domain.UserIDPLink, len(idps))
|
|
for i, idp := range idps {
|
|
externalIDPs[i] = &domain.UserIDPLink{
|
|
ObjectRoot: models.ObjectRoot{
|
|
AggregateID: idp.UserID,
|
|
ResourceOwner: idp.ResourceOwner,
|
|
},
|
|
IDPConfigID: idp.IDPID,
|
|
ExternalUserID: idp.ProvidedUserID,
|
|
DisplayName: idp.ProvidedUsername,
|
|
}
|
|
}
|
|
return externalIDPs
|
|
}
|
|
|
|
func listProvidersToQuery(instanceID string, req *admin_pb.ListProvidersRequest) (*query.IDPTemplateSearchQueries, error) {
|
|
offset, limit, asc := object.ListQueryToModel(req.Query)
|
|
queries, err := providerQueriesToQuery(req.Queries)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
iamQuery, err := query.NewIDPTemplateResourceOwnerSearchQuery(instanceID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
queries = append(queries, iamQuery)
|
|
return &query.IDPTemplateSearchQueries{
|
|
SearchRequest: query.SearchRequest{
|
|
Offset: offset,
|
|
Limit: limit,
|
|
Asc: asc,
|
|
},
|
|
Queries: queries,
|
|
}, nil
|
|
}
|
|
|
|
func providerQueriesToQuery(queries []*admin_pb.ProviderQuery) (q []query.SearchQuery, err error) {
|
|
q = make([]query.SearchQuery, len(queries))
|
|
for i, query := range queries {
|
|
q[i], err = providerQueryToQuery(query)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return q, nil
|
|
}
|
|
|
|
func providerQueryToQuery(idpQuery *admin_pb.ProviderQuery) (query.SearchQuery, error) {
|
|
switch q := idpQuery.Query.(type) {
|
|
case *admin_pb.ProviderQuery_IdpNameQuery:
|
|
return query.NewIDPTemplateNameSearchQuery(object.TextMethodToQuery(q.IdpNameQuery.Method), q.IdpNameQuery.Name)
|
|
case *admin_pb.ProviderQuery_IdpIdQuery:
|
|
return query.NewIDPTemplateIDSearchQuery(q.IdpIdQuery.Id)
|
|
default:
|
|
return nil, zerrors.ThrowInvalidArgument(nil, "ADMIN-Dr2aa", "List.Query.Invalid")
|
|
}
|
|
}
|
|
|
|
func addGenericOAuthProviderToCommand(req *admin_pb.AddGenericOAuthProviderRequest) command.GenericOAuthProvider {
|
|
return command.GenericOAuthProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
AuthorizationEndpoint: req.AuthorizationEndpoint,
|
|
TokenEndpoint: req.TokenEndpoint,
|
|
UserEndpoint: req.UserEndpoint,
|
|
Scopes: req.Scopes,
|
|
IDAttribute: req.IdAttribute,
|
|
UsePKCE: req.UsePkce,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGenericOAuthProviderToCommand(req *admin_pb.UpdateGenericOAuthProviderRequest) command.GenericOAuthProvider {
|
|
return command.GenericOAuthProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
AuthorizationEndpoint: req.AuthorizationEndpoint,
|
|
TokenEndpoint: req.TokenEndpoint,
|
|
UserEndpoint: req.UserEndpoint,
|
|
Scopes: req.Scopes,
|
|
IDAttribute: req.IdAttribute,
|
|
UsePKCE: req.UsePkce,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGenericOIDCProviderToCommand(req *admin_pb.AddGenericOIDCProviderRequest) command.GenericOIDCProvider {
|
|
return command.GenericOIDCProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IsIDTokenMapping: req.IsIdTokenMapping,
|
|
UsePKCE: req.UsePkce,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGenericOIDCProviderToCommand(req *admin_pb.UpdateGenericOIDCProviderRequest) command.GenericOIDCProvider {
|
|
return command.GenericOIDCProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IsIDTokenMapping: req.IsIdTokenMapping,
|
|
UsePKCE: req.UsePkce,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addJWTProviderToCommand(req *admin_pb.AddJWTProviderRequest) command.JWTProvider {
|
|
return command.JWTProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
JWTEndpoint: req.JwtEndpoint,
|
|
KeyEndpoint: req.KeysEndpoint,
|
|
HeaderName: req.HeaderName,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateJWTProviderToCommand(req *admin_pb.UpdateJWTProviderRequest) command.JWTProvider {
|
|
return command.JWTProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
JWTEndpoint: req.JwtEndpoint,
|
|
KeyEndpoint: req.KeysEndpoint,
|
|
HeaderName: req.HeaderName,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addAzureADProviderToCommand(req *admin_pb.AddAzureADProviderRequest) command.AzureADProvider {
|
|
return command.AzureADProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
Tenant: idp_grpc.AzureADTenantToCommand(req.Tenant),
|
|
EmailVerified: req.EmailVerified,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateAzureADProviderToCommand(req *admin_pb.UpdateAzureADProviderRequest) command.AzureADProvider {
|
|
return command.AzureADProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
Tenant: idp_grpc.AzureADTenantToCommand(req.Tenant),
|
|
EmailVerified: req.EmailVerified,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGitHubProviderToCommand(req *admin_pb.AddGitHubProviderRequest) command.GitHubProvider {
|
|
return command.GitHubProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGitHubProviderToCommand(req *admin_pb.UpdateGitHubProviderRequest) command.GitHubProvider {
|
|
return command.GitHubProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGitHubEnterpriseProviderToCommand(req *admin_pb.AddGitHubEnterpriseServerProviderRequest) command.GitHubEnterpriseProvider {
|
|
return command.GitHubEnterpriseProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
AuthorizationEndpoint: req.AuthorizationEndpoint,
|
|
TokenEndpoint: req.TokenEndpoint,
|
|
UserEndpoint: req.UserEndpoint,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGitHubEnterpriseProviderToCommand(req *admin_pb.UpdateGitHubEnterpriseServerProviderRequest) command.GitHubEnterpriseProvider {
|
|
return command.GitHubEnterpriseProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
AuthorizationEndpoint: req.AuthorizationEndpoint,
|
|
TokenEndpoint: req.TokenEndpoint,
|
|
UserEndpoint: req.UserEndpoint,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGitLabProviderToCommand(req *admin_pb.AddGitLabProviderRequest) command.GitLabProvider {
|
|
return command.GitLabProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGitLabProviderToCommand(req *admin_pb.UpdateGitLabProviderRequest) command.GitLabProvider {
|
|
return command.GitLabProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGitLabSelfHostedProviderToCommand(req *admin_pb.AddGitLabSelfHostedProviderRequest) command.GitLabSelfHostedProvider {
|
|
return command.GitLabSelfHostedProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGitLabSelfHostedProviderToCommand(req *admin_pb.UpdateGitLabSelfHostedProviderRequest) command.GitLabSelfHostedProvider {
|
|
return command.GitLabSelfHostedProvider{
|
|
Name: req.Name,
|
|
Issuer: req.Issuer,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addGoogleProviderToCommand(req *admin_pb.AddGoogleProviderRequest) command.GoogleProvider {
|
|
return command.GoogleProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateGoogleProviderToCommand(req *admin_pb.UpdateGoogleProviderRequest) command.GoogleProvider {
|
|
return command.GoogleProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
ClientSecret: req.ClientSecret,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addLDAPProviderToCommand(req *admin_pb.AddLDAPProviderRequest) command.LDAPProvider {
|
|
return command.LDAPProvider{
|
|
Name: req.Name,
|
|
Servers: req.Servers,
|
|
StartTLS: req.StartTls,
|
|
BaseDN: req.BaseDn,
|
|
BindDN: req.BindDn,
|
|
BindPassword: req.BindPassword,
|
|
UserBase: req.UserBase,
|
|
UserObjectClasses: req.UserObjectClasses,
|
|
UserFilters: req.UserFilters,
|
|
Timeout: req.Timeout.AsDuration(),
|
|
RootCA: req.RootCa,
|
|
LDAPAttributes: idp_grpc.LDAPAttributesToCommand(req.Attributes),
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateLDAPProviderToCommand(req *admin_pb.UpdateLDAPProviderRequest) command.LDAPProvider {
|
|
return command.LDAPProvider{
|
|
Name: req.Name,
|
|
Servers: req.Servers,
|
|
StartTLS: req.StartTls,
|
|
BaseDN: req.BaseDn,
|
|
BindDN: req.BindDn,
|
|
BindPassword: req.BindPassword,
|
|
UserBase: req.UserBase,
|
|
UserObjectClasses: req.UserObjectClasses,
|
|
UserFilters: req.UserFilters,
|
|
Timeout: req.Timeout.AsDuration(),
|
|
LDAPAttributes: idp_grpc.LDAPAttributesToCommand(req.Attributes),
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
RootCA: req.RootCa,
|
|
}
|
|
}
|
|
|
|
func addAppleProviderToCommand(req *admin_pb.AddAppleProviderRequest) command.AppleProvider {
|
|
return command.AppleProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
TeamID: req.TeamId,
|
|
KeyID: req.KeyId,
|
|
PrivateKey: req.PrivateKey,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateAppleProviderToCommand(req *admin_pb.UpdateAppleProviderRequest) command.AppleProvider {
|
|
return command.AppleProvider{
|
|
Name: req.Name,
|
|
ClientID: req.ClientId,
|
|
TeamID: req.TeamId,
|
|
KeyID: req.KeyId,
|
|
PrivateKey: req.PrivateKey,
|
|
Scopes: req.Scopes,
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func addSAMLProviderToCommand(req *admin_pb.AddSAMLProviderRequest) *command.SAMLProvider {
|
|
var nameIDFormat *domain.SAMLNameIDFormat
|
|
if req.NameIdFormat != nil {
|
|
nameIDFormat = gu.Ptr(idp_grpc.SAMLNameIDFormatToDomain(req.GetNameIdFormat()))
|
|
}
|
|
return &command.SAMLProvider{
|
|
Name: req.Name,
|
|
Metadata: req.GetMetadataXml(),
|
|
MetadataURL: req.GetMetadataUrl(),
|
|
Binding: bindingToCommand(req.Binding),
|
|
WithSignedRequest: req.WithSignedRequest,
|
|
NameIDFormat: nameIDFormat,
|
|
TransientMappingAttributeName: req.GetTransientMappingAttributeName(),
|
|
FederatedLogoutEnabled: req.GetFederatedLogoutEnabled(),
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func updateSAMLProviderToCommand(req *admin_pb.UpdateSAMLProviderRequest) *command.SAMLProvider {
|
|
var nameIDFormat *domain.SAMLNameIDFormat
|
|
if req.NameIdFormat != nil {
|
|
nameIDFormat = gu.Ptr(idp_grpc.SAMLNameIDFormatToDomain(req.GetNameIdFormat()))
|
|
}
|
|
return &command.SAMLProvider{
|
|
Name: req.Name,
|
|
Metadata: req.GetMetadataXml(),
|
|
MetadataURL: req.GetMetadataUrl(),
|
|
Binding: bindingToCommand(req.Binding),
|
|
WithSignedRequest: req.WithSignedRequest,
|
|
NameIDFormat: nameIDFormat,
|
|
TransientMappingAttributeName: req.GetTransientMappingAttributeName(),
|
|
FederatedLogoutEnabled: req.GetFederatedLogoutEnabled(),
|
|
IDPOptions: idp_grpc.OptionsToCommand(req.ProviderOptions),
|
|
}
|
|
}
|
|
|
|
func bindingToCommand(binding idp_pb.SAMLBinding) string {
|
|
switch binding {
|
|
case idp_pb.SAMLBinding_SAML_BINDING_UNSPECIFIED:
|
|
return ""
|
|
case idp_pb.SAMLBinding_SAML_BINDING_POST:
|
|
return saml.HTTPPostBinding
|
|
case idp_pb.SAMLBinding_SAML_BINDING_REDIRECT:
|
|
return saml.HTTPRedirectBinding
|
|
case idp_pb.SAMLBinding_SAML_BINDING_ARTIFACT:
|
|
return saml.HTTPArtifactBinding
|
|
default:
|
|
return ""
|
|
}
|
|
}
|