mirror of
				https://github.com/zitadel/zitadel.git
				synced 2025-10-25 20:38:48 +00:00 
			
		
		
		
	feat: add SAML as identity provider (#6454)
* feat: first implementation for saml sp * fix: add command side instance and org for saml provider * fix: add query side instance and org for saml provider * fix: request handling in event and retrieval of finished intent * fix: add review changes and integration tests * fix: add integration tests for saml idp * fix: correct unit tests with review changes * fix: add saml session unit test * fix: add saml session unit test * fix: add saml session unit test * fix: changes from review * fix: changes from review * fix: proto build error * fix: proto build error * fix: proto build error * fix: proto require metadata oneof * fix: login with saml provider * fix: integration test for saml assertion * lint client.go * fix json tag * fix: linting * fix import * fix: linting * fix saml idp query * fix: linting * lint: try all issues * revert linting config * fix: add regenerate endpoints * fix: translations * fix mk.yaml * ignore acs path for user agent cookie * fix: add AuthFromProvider test for saml * fix: integration test for saml retrieve information --------- Co-authored-by: Livio Spring <livio.a@gmail.com>
This commit is contained in:
		| @@ -5318,3 +5318,527 @@ func TestCommandSide_UpdateInstanceAppleIDP(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCommandSide_AddInstanceSAMLIDP(t *testing.T) { | ||||
| 	type fields struct { | ||||
| 		eventstore                 *eventstore.Eventstore | ||||
| 		idGenerator                id.Generator | ||||
| 		secretCrypto               crypto.EncryptionAlgorithm | ||||
| 		certificateAndKeyGenerator func(id string) ([]byte, []byte, error) | ||||
| 	} | ||||
| 	type args struct { | ||||
| 		ctx      context.Context | ||||
| 		provider SAMLProvider | ||||
| 	} | ||||
| 	type res struct { | ||||
| 		id   string | ||||
| 		want *domain.ObjectDetails | ||||
| 		err  func(error) bool | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		name   string | ||||
| 		fields fields | ||||
| 		args   args | ||||
| 		res    res | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"invalid name", | ||||
| 			fields{ | ||||
| 				eventstore:  eventstoreExpect(t), | ||||
| 				idGenerator: id_mock.NewIDGeneratorExpectIDs(t, "id1"), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx:      authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				provider: SAMLProvider{}, | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-o07zjotgnd", "")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"invalid metadata", | ||||
| 			fields{ | ||||
| 				eventstore:  eventstoreExpect(t), | ||||
| 				idGenerator: id_mock.NewIDGeneratorExpectIDs(t, "id1"), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name: "name", | ||||
| 				}, | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-3bi3esi16t", "Errors.Invalid.Argument")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "ok", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter(), | ||||
| 					expectPush( | ||||
| 						[]*repository.Event{ | ||||
| 							eventFromEventPusherWithInstanceID( | ||||
| 								"instance1", | ||||
| 								instance.NewSAMLIDPAddedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 									"id1", | ||||
| 									"name", | ||||
| 									[]byte("metadata"), | ||||
| 									&crypto.CryptoValue{ | ||||
| 										CryptoType: crypto.TypeEncryption, | ||||
| 										Algorithm:  "enc", | ||||
| 										KeyID:      "id", | ||||
| 										Crypted:    []byte("key"), | ||||
| 									}, | ||||
| 									[]byte("certificate"), | ||||
| 									"", | ||||
| 									false, | ||||
| 									idp.Options{}, | ||||
| 								)), | ||||
| 						}, | ||||
| 					), | ||||
| 				), | ||||
| 				idGenerator:                id_mock.NewIDGeneratorExpectIDs(t, "id1"), | ||||
| 				secretCrypto:               crypto.CreateMockEncryptionAlg(gomock.NewController(t)), | ||||
| 				certificateAndKeyGenerator: func(id string) ([]byte, []byte, error) { return []byte("key"), []byte("certificate"), nil }, | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name:     "name", | ||||
| 					Metadata: []byte("metadata"), | ||||
| 				}, | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				id:   "id1", | ||||
| 				want: &domain.ObjectDetails{ResourceOwner: "instance1"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "ok all set", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter(), | ||||
| 					expectPush( | ||||
| 						[]*repository.Event{ | ||||
| 							eventFromEventPusherWithInstanceID( | ||||
| 								"instance1", | ||||
| 								instance.NewSAMLIDPAddedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 									"id1", | ||||
| 									"name", | ||||
| 									[]byte("metadata"), | ||||
| 									&crypto.CryptoValue{ | ||||
| 										CryptoType: crypto.TypeEncryption, | ||||
| 										Algorithm:  "enc", | ||||
| 										KeyID:      "id", | ||||
| 										Crypted:    []byte("key"), | ||||
| 									}, | ||||
| 									[]byte("certificate"), | ||||
| 									"binding", | ||||
| 									true, | ||||
| 									idp.Options{ | ||||
| 										IsCreationAllowed: true, | ||||
| 										IsLinkingAllowed:  true, | ||||
| 										IsAutoCreation:    true, | ||||
| 										IsAutoUpdate:      true, | ||||
| 									}, | ||||
| 								)), | ||||
| 						}, | ||||
| 					), | ||||
| 				), | ||||
| 				idGenerator:                id_mock.NewIDGeneratorExpectIDs(t, "id1"), | ||||
| 				secretCrypto:               crypto.CreateMockEncryptionAlg(gomock.NewController(t)), | ||||
| 				certificateAndKeyGenerator: func(id string) ([]byte, []byte, error) { return []byte("key"), []byte("certificate"), nil }, | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name:              "name", | ||||
| 					Metadata:          []byte("metadata"), | ||||
| 					Binding:           "binding", | ||||
| 					WithSignedRequest: true, | ||||
| 					IDPOptions: idp.Options{ | ||||
| 						IsCreationAllowed: true, | ||||
| 						IsLinkingAllowed:  true, | ||||
| 						IsAutoCreation:    true, | ||||
| 						IsAutoUpdate:      true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				id:   "id1", | ||||
| 				want: &domain.ObjectDetails{ResourceOwner: "instance1"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			c := &Commands{ | ||||
| 				eventstore:                     tt.fields.eventstore, | ||||
| 				idGenerator:                    tt.fields.idGenerator, | ||||
| 				idpConfigEncryption:            tt.fields.secretCrypto, | ||||
| 				samlCertificateAndKeyGenerator: tt.fields.certificateAndKeyGenerator, | ||||
| 			} | ||||
| 			id, got, err := c.AddInstanceSAMLProvider(tt.args.ctx, tt.args.provider) | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.NoError(t, err) | ||||
| 			} | ||||
| 			if tt.res.err != nil && !tt.res.err(err) { | ||||
| 				t.Errorf("got wrong err: %v ", err) | ||||
| 			} | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.Equal(t, tt.res.id, id) | ||||
| 				assert.Equal(t, tt.res.want, got) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCommandSide_UpdateInstanceGenericSAMLIDP(t *testing.T) { | ||||
| 	type fields struct { | ||||
| 		eventstore   *eventstore.Eventstore | ||||
| 		secretCrypto crypto.EncryptionAlgorithm | ||||
| 	} | ||||
| 	type args struct { | ||||
| 		ctx      context.Context | ||||
| 		id       string | ||||
| 		provider SAMLProvider | ||||
| 	} | ||||
| 	type res struct { | ||||
| 		want *domain.ObjectDetails | ||||
| 		err  func(error) bool | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		name   string | ||||
| 		fields fields | ||||
| 		args   args | ||||
| 		res    res | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"invalid id", | ||||
| 			fields{ | ||||
| 				eventstore: eventstoreExpect(t), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx:      authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				provider: SAMLProvider{}, | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-7o3rq1owpm", "")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"invalid name", | ||||
| 			fields{ | ||||
| 				eventstore: eventstoreExpect(t), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx:      authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:       "id1", | ||||
| 				provider: SAMLProvider{}, | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-q2s9rak7o9", "")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"invalid metadata", | ||||
| 			fields{ | ||||
| 				eventstore: eventstoreExpect(t), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name: "name", | ||||
| 				}, | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-iw1rxnf4sf", "")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "not found", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter(), | ||||
| 				), | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name:     "name", | ||||
| 					Metadata: []byte("metadata"), | ||||
| 				}, | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				err: caos_errors.IsNotFound, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "no changes", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter( | ||||
| 						eventFromEventPusher( | ||||
| 							instance.NewSAMLIDPAddedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 								"id1", | ||||
| 								"name", | ||||
| 								[]byte("metadata"), | ||||
| 								&crypto.CryptoValue{ | ||||
| 									CryptoType: crypto.TypeEncryption, | ||||
| 									Algorithm:  "enc", | ||||
| 									KeyID:      "id", | ||||
| 									Crypted:    []byte("key"), | ||||
| 								}, | ||||
| 								[]byte("certificate"), | ||||
| 								"", | ||||
| 								false, | ||||
| 								idp.Options{}, | ||||
| 							)), | ||||
| 					), | ||||
| 				), | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name:     "name", | ||||
| 					Metadata: []byte("metadata"), | ||||
| 				}, | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				want: &domain.ObjectDetails{ResourceOwner: "instance1"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "change ok", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter( | ||||
| 						eventFromEventPusher( | ||||
| 							instance.NewSAMLIDPAddedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 								"id1", | ||||
| 								"name", | ||||
| 								[]byte("metadata"), | ||||
| 								&crypto.CryptoValue{ | ||||
| 									CryptoType: crypto.TypeEncryption, | ||||
| 									Algorithm:  "enc", | ||||
| 									KeyID:      "id", | ||||
| 									Crypted:    []byte("key"), | ||||
| 								}, | ||||
| 								[]byte("certificate"), | ||||
| 								"binding", | ||||
| 								false, | ||||
| 								idp.Options{}, | ||||
| 							)), | ||||
| 					), | ||||
| 					expectPush( | ||||
| 						[]*repository.Event{ | ||||
| 							eventFromEventPusherWithInstanceID( | ||||
| 								"instance1", | ||||
| 								func() eventstore.Command { | ||||
| 									t := true | ||||
| 									event, _ := instance.NewSAMLIDPChangedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 										"id1", | ||||
| 										[]idp.SAMLIDPChanges{ | ||||
| 											idp.ChangeSAMLName("new name"), | ||||
| 											idp.ChangeSAMLMetadata([]byte("new metadata")), | ||||
| 											idp.ChangeSAMLBinding("new binding"), | ||||
| 											idp.ChangeSAMLWithSignedRequest(true), | ||||
| 											idp.ChangeSAMLOptions(idp.OptionChanges{ | ||||
| 												IsCreationAllowed: &t, | ||||
| 												IsLinkingAllowed:  &t, | ||||
| 												IsAutoCreation:    &t, | ||||
| 												IsAutoUpdate:      &t, | ||||
| 											}), | ||||
| 										}, | ||||
| 									) | ||||
| 									return event | ||||
| 								}(), | ||||
| 							), | ||||
| 						}, | ||||
| 					), | ||||
| 				), | ||||
| 				secretCrypto: crypto.CreateMockEncryptionAlg(gomock.NewController(t)), | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 				provider: SAMLProvider{ | ||||
| 					Name:              "new name", | ||||
| 					Metadata:          []byte("new metadata"), | ||||
| 					Binding:           "new binding", | ||||
| 					WithSignedRequest: true, | ||||
| 					IDPOptions: idp.Options{ | ||||
| 						IsCreationAllowed: true, | ||||
| 						IsLinkingAllowed:  true, | ||||
| 						IsAutoCreation:    true, | ||||
| 						IsAutoUpdate:      true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				want: &domain.ObjectDetails{ResourceOwner: "instance1"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			c := &Commands{ | ||||
| 				eventstore:          tt.fields.eventstore, | ||||
| 				idpConfigEncryption: tt.fields.secretCrypto, | ||||
| 			} | ||||
| 			got, err := c.UpdateInstanceSAMLProvider(tt.args.ctx, tt.args.id, tt.args.provider) | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.NoError(t, err) | ||||
| 			} | ||||
| 			if tt.res.err != nil && !tt.res.err(err) { | ||||
| 				t.Errorf("got wrong err: %v ", err) | ||||
| 			} | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.Equal(t, tt.res.want, got) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCommandSide_RegenerateInstanceSAMLProviderCertificate(t *testing.T) { | ||||
| 	type fields struct { | ||||
| 		eventstore                 *eventstore.Eventstore | ||||
| 		secretCrypto               crypto.EncryptionAlgorithm | ||||
| 		certificateAndKeyGenerator func(id string) ([]byte, []byte, error) | ||||
| 	} | ||||
| 	type args struct { | ||||
| 		ctx context.Context | ||||
| 		id  string | ||||
| 	} | ||||
| 	type res struct { | ||||
| 		want *domain.ObjectDetails | ||||
| 		err  func(error) bool | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		name   string | ||||
| 		fields fields | ||||
| 		args   args | ||||
| 		res    res | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"invalid id", | ||||
| 			fields{ | ||||
| 				eventstore: eventstoreExpect(t), | ||||
| 			}, | ||||
| 			args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 			}, | ||||
| 			res{ | ||||
| 				err: func(err error) bool { | ||||
| 					return errors.Is(err, caos_errors.ThrowInvalidArgument(nil, "INST-7de108gqya", "")) | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "not found", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter(), | ||||
| 				), | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				err: caos_errors.IsNotFound, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "change ok", | ||||
| 			fields: fields{ | ||||
| 				eventstore: eventstoreExpect(t, | ||||
| 					expectFilter( | ||||
| 						eventFromEventPusher( | ||||
| 							instance.NewSAMLIDPAddedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 								"id1", | ||||
| 								"name", | ||||
| 								[]byte("metadata"), | ||||
| 								&crypto.CryptoValue{ | ||||
| 									CryptoType: crypto.TypeEncryption, | ||||
| 									Algorithm:  "enc", | ||||
| 									KeyID:      "id", | ||||
| 									Crypted:    []byte("key"), | ||||
| 								}, | ||||
| 								[]byte("certificate"), | ||||
| 								"binding", | ||||
| 								false, | ||||
| 								idp.Options{}, | ||||
| 							)), | ||||
| 					), | ||||
| 					expectPush( | ||||
| 						[]*repository.Event{ | ||||
| 							eventFromEventPusherWithInstanceID( | ||||
| 								"instance1", | ||||
| 								func() eventstore.Command { | ||||
| 									event, _ := instance.NewSAMLIDPChangedEvent(context.Background(), &instance.NewAggregate("instance1").Aggregate, | ||||
| 										"id1", | ||||
| 										[]idp.SAMLIDPChanges{ | ||||
| 											idp.ChangeSAMLKey(&crypto.CryptoValue{ | ||||
| 												CryptoType: crypto.TypeEncryption, | ||||
| 												Algorithm:  "enc", | ||||
| 												KeyID:      "id", | ||||
| 												Crypted:    []byte("new key"), | ||||
| 											}), | ||||
| 											idp.ChangeSAMLCertificate([]byte("new certificate")), | ||||
| 										}, | ||||
| 									) | ||||
| 									return event | ||||
| 								}(), | ||||
| 							), | ||||
| 						}, | ||||
| 					), | ||||
| 				), | ||||
| 				secretCrypto: crypto.CreateMockEncryptionAlg(gomock.NewController(t)), | ||||
| 				certificateAndKeyGenerator: func(id string) ([]byte, []byte, error) { | ||||
| 					return []byte("new key"), []byte("new certificate"), nil | ||||
| 				}, | ||||
| 			}, | ||||
| 			args: args{ | ||||
| 				ctx: authz.WithInstanceID(context.Background(), "instance1"), | ||||
| 				id:  "id1", | ||||
| 			}, | ||||
| 			res: res{ | ||||
| 				want: &domain.ObjectDetails{ResourceOwner: "instance1"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			c := &Commands{ | ||||
| 				eventstore:                     tt.fields.eventstore, | ||||
| 				idpConfigEncryption:            tt.fields.secretCrypto, | ||||
| 				samlCertificateAndKeyGenerator: tt.fields.certificateAndKeyGenerator, | ||||
| 			} | ||||
| 			got, err := c.RegenerateInstanceSAMLProviderCertificate(tt.args.ctx, tt.args.id) | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.NoError(t, err) | ||||
| 			} | ||||
| 			if tt.res.err != nil && !tt.res.err(err) { | ||||
| 				t.Errorf("got wrong err: %v ", err) | ||||
| 			} | ||||
| 			if tt.res.err == nil { | ||||
| 				assert.Equal(t, tt.res.want, got) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Stefan Benz
					Stefan Benz