Skip to content

Commit abc5479

Browse files
Add support for SSL bundles in SAML2 signing auto-configuration
1 parent 14edcb9 commit abc5479

File tree

9 files changed

+462
-111
lines changed

9 files changed

+462
-111
lines changed

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyProperties.java

+9-38
Original file line numberDiff line numberDiff line change
@@ -178,7 +178,7 @@ public static class Credential {
178178
* SSL bundle providing a private key used for signing and a Relying Party
179179
* X509Certificate shared with the identity provider.
180180
*/
181-
private Bundle bundle;
181+
private String bundle;
182182

183183
public Resource getPrivateKeyLocation() {
184184
return this.privateKeyLocation;
@@ -196,11 +196,11 @@ public void setCertificateLocation(Resource certificate) {
196196
this.certificateLocation = certificate;
197197
}
198198

199-
public Bundle getBundle() {
199+
public String getBundle() {
200200
return this.bundle;
201201
}
202202

203-
public void setBundle(Bundle bundle) {
203+
public void setBundle(String bundle) {
204204
this.bundle = bundle;
205205
}
206206

@@ -241,7 +241,7 @@ public static class Credential {
241241
* SSL bundle providing a private key used for decrypting and a Relying Party
242242
* X509Certificate shared with the identity provider.
243243
*/
244-
private Bundle bundle;
244+
private String bundle;
245245

246246
public Resource getPrivateKeyLocation() {
247247
return this.privateKeyLocation;
@@ -259,11 +259,11 @@ public void setCertificateLocation(Resource certificate) {
259259
this.certificateLocation = certificate;
260260
}
261261

262-
public Bundle getBundle() {
262+
public String getBundle() {
263263
return this.bundle;
264264
}
265265

266-
public void setBundle(Bundle bundle) {
266+
public void setBundle(String bundle) {
267267
this.bundle = bundle;
268268
}
269269

@@ -400,7 +400,7 @@ public static class Credential {
400400
* SSL bundle providing the X.509 certificate used for verification of
401401
* incoming SAML messages.
402402
*/
403-
private Bundle bundle;
403+
private String bundle;
404404

405405
public Resource getCertificateLocation() {
406406
return this.certificate;
@@ -410,11 +410,11 @@ public void setCertificateLocation(Resource certificate) {
410410
this.certificate = certificate;
411411
}
412412

413-
public Bundle getBundle() {
413+
public String getBundle() {
414414
return this.bundle;
415415
}
416416

417-
public void setBundle(Bundle bundle) {
417+
public void setBundle(String bundle) {
418418
this.bundle = bundle;
419419
}
420420

@@ -470,33 +470,4 @@ public void setBinding(Saml2MessageBinding binding) {
470470

471471
}
472472

473-
public static class Bundle {
474-
475-
/**
476-
* Name of the SSL bundle.
477-
*/
478-
private String name;
479-
480-
/**
481-
* Alias for the certificate to use from the SSL bundle.
482-
*/
483-
private String alias;
484-
485-
public String getName() {
486-
return this.name;
487-
}
488-
489-
public void setName(String name) {
490-
this.name = name;
491-
}
492-
493-
public String getAlias() {
494-
return this.alias;
495-
}
496-
497-
public void setAlias(String alias) {
498-
this.alias = alias;
499-
}
500-
}
501-
502473
}

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyRegistrationConfiguration.java

+18-37
Original file line numberDiff line numberDiff line change
@@ -17,11 +17,7 @@
1717
package org.springframework.boot.autoconfigure.security.saml2;
1818

1919
import java.io.InputStream;
20-
import java.security.GeneralSecurityException;
21-
import java.security.Key;
22-
import java.security.KeyStore;
2320
import java.security.PrivateKey;
24-
import java.security.cert.Certificate;
2521
import java.security.cert.CertificateFactory;
2622
import java.security.cert.X509Certificate;
2723
import java.security.interfaces.RSAPrivateKey;
@@ -34,14 +30,14 @@
3430
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
3531
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.AssertingParty;
3632
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.AssertingParty.Verification;
37-
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.Bundle;
3833
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.Decryption;
3934
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.Registration;
4035
import org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyProperties.Registration.Signing.Credential;
4136
import org.springframework.boot.context.properties.PropertyMapper;
4237
import org.springframework.boot.ssl.SslBundle;
43-
import org.springframework.boot.ssl.SslBundleKey;
38+
import org.springframework.boot.ssl.SslBundleKeyStore;
4439
import org.springframework.boot.ssl.SslBundles;
40+
import org.springframework.boot.ssl.SslStoreBundle;
4541
import org.springframework.context.annotation.Bean;
4642
import org.springframework.context.annotation.Conditional;
4743
import org.springframework.context.annotation.Configuration;
@@ -163,9 +159,9 @@ private void validateSigningCredentials(Registration properties, boolean signReq
163159
}
164160

165161
private Saml2X509Credential asSigningCredential(Credential properties, SslBundles sslBundles) {
166-
Bundle sslBundle = properties.getBundle();
162+
String sslBundle = properties.getBundle();
167163
if (sslBundle != null) {
168-
PrivateKey privateKey = getPrivateKey(sslBundle.getName(), sslBundles);
164+
PrivateKey privateKey = getPrivateKey(sslBundle, sslBundles);
169165
X509Certificate certificate = getCertificate(sslBundle, sslBundles);
170166
return new Saml2X509Credential(privateKey, certificate, Saml2X509CredentialType.SIGNING);
171167
}
@@ -175,9 +171,9 @@ private Saml2X509Credential asSigningCredential(Credential properties, SslBundle
175171
}
176172

177173
private Saml2X509Credential asDecryptionCredential(Decryption.Credential properties, SslBundles sslBundles) {
178-
Bundle sslBundle = properties.getBundle();
174+
String sslBundle = properties.getBundle();
179175
if (sslBundle != null) {
180-
PrivateKey privateKey = getPrivateKey(sslBundle.getName(), sslBundles);
176+
PrivateKey privateKey = getPrivateKey(sslBundle, sslBundles);
181177
X509Certificate certificate = getCertificate(sslBundle, sslBundles);
182178
return new Saml2X509Credential(privateKey, certificate, Saml2X509CredentialType.DECRYPTION);
183179
}
@@ -187,7 +183,7 @@ private Saml2X509Credential asDecryptionCredential(Decryption.Credential propert
187183
}
188184

189185
private Saml2X509Credential asVerificationCredential(Verification.Credential properties, SslBundles sslBundles) {
190-
Bundle sslBundle = properties.getBundle();
186+
String sslBundle = properties.getBundle();
191187
if (sslBundle != null) {
192188
X509Certificate certificate = getCertificate(sslBundle, sslBundles);
193189
return new Saml2X509Credential(certificate, Saml2X509Credential.Saml2X509CredentialType.ENCRYPTION,
@@ -221,34 +217,19 @@ private X509Certificate readCertificate(Resource location) {
221217
}
222218

223219
private PrivateKey getPrivateKey(String sslBundle, SslBundles sslBundles) {
224-
try {
225-
SslBundle bundle = sslBundles.getBundle(sslBundle);
226-
SslBundleKey key = bundle.getKey();
227-
KeyStore keyStore = bundle.getStores().getKeyStore();
228-
Key privateKey = keyStore.getKey(key.getAlias(), key.getPassword().toCharArray());
229-
Assert.notNull(privateKey,
230-
"Private key with alias '" + key.getAlias() + "' was not found in SSL bundle '" + sslBundle + "'");
231-
Assert.isInstanceOf(PrivateKey.class, privateKey);
232-
return (PrivateKey) privateKey;
233-
}
234-
catch (GeneralSecurityException ex) {
235-
throw new IllegalStateException("Error getting private key from SSL bundle '" + sslBundle + "'", ex);
236-
}
220+
SslBundle bundle = sslBundles.getBundle(sslBundle);
221+
SslStoreBundle stores = bundle.getStores();
222+
PrivateKey privateKey = SslBundleKeyStore.from(stores.getKeyStore(), bundle.getKey()).getPrivateKey();
223+
Assert.notNull(privateKey, "KeyStore in SSL bundle '" + sslBundle + "' must have a private key");
224+
return privateKey;
237225
}
238226

239-
private X509Certificate getCertificate(Bundle sslBundle, SslBundles sslBundles) {
240-
try {
241-
SslBundle bundle = sslBundles.getBundle(sslBundle.getName());
242-
KeyStore keyStore = bundle.getStores().getKeyStore();
243-
Certificate certificate = keyStore.getCertificate(sslBundle.getAlias());
244-
Assert.notNull(certificate, "Certificate with alias '" + sslBundle.getAlias()
245-
+ "' was not found in SSL bundle '" + sslBundle + "'");
246-
Assert.isInstanceOf(X509Certificate.class, certificate);
247-
return (X509Certificate) certificate;
248-
}
249-
catch (GeneralSecurityException ex) {
250-
throw new IllegalStateException("Error getting certificate from SSL bundle '" + sslBundle + "'", ex);
251-
}
227+
private X509Certificate getCertificate(String sslBundle, SslBundles sslBundles) {
228+
SslBundle bundle = sslBundles.getBundle(sslBundle);
229+
SslStoreBundle stores = bundle.getStores();
230+
X509Certificate certificate = SslBundleKeyStore.from(stores.getKeyStore(), bundle.getKey()).getCertificate();
231+
Assert.notNull(certificate, "KeyStore in SSL bundle '" + sslBundle + "' must have a certificate");
232+
return certificate;
252233
}
253234

254235
}

spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/security/saml2/Saml2RelyingPartyAutoConfigurationTests.java

+6-8
Original file line numberDiff line numberDiff line change
@@ -65,8 +65,9 @@ class Saml2RelyingPartyAutoConfigurationTests {
6565

6666
private static final String PREFIX = "spring.security.saml2.relyingparty.registration";
6767

68-
private final WebApplicationContextRunner contextRunner = new WebApplicationContextRunner().withConfiguration(
69-
AutoConfigurations.of(Saml2RelyingPartyAutoConfiguration.class, SecurityAutoConfiguration.class, SslAutoConfiguration.class));
68+
private final WebApplicationContextRunner contextRunner = new WebApplicationContextRunner()
69+
.withConfiguration(AutoConfigurations.of(Saml2RelyingPartyAutoConfiguration.class,
70+
SecurityAutoConfiguration.class, SslAutoConfiguration.class));
7071

7172
@Test
7273
void autoConfigurationShouldBeConditionalOnRelyingPartyRegistrationRepositoryClass() {
@@ -366,19 +367,16 @@ private String[] getPropertyValuesWithSslBundles() {
366367
"spring.ssl.bundle.pem.saml.key.password=secret1",
367368
"spring.ssl.bundle.pem.saml.keystore.certificate=classpath:saml/certificate-location",
368369
"spring.ssl.bundle.pem.saml.keystore.private-key=classpath:saml/private-key-location",
369-
PREFIX + ".foo.signing.credentials[0].bundle.name=saml",
370-
PREFIX + ".foo.signing.credentials[0].bundle.alias=key-alias",
371-
PREFIX + ".foo.decryption.credentials[0].bundle.name=saml",
372-
PREFIX + ".foo.decryption.credentials[0].bundle.alias=key-alias",
370+
PREFIX + ".foo.signing.credentials[0].bundle=saml",
371+
PREFIX + ".foo.decryption.credentials[0].bundle=saml",
373372
PREFIX + ".foo.singlelogout.url=https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/SLOService.php",
374373
PREFIX + ".foo.singlelogout.response-url=https://simplesaml-for-spring-saml.cfapps.io/",
375374
PREFIX + ".foo.singlelogout.binding=post",
376375
PREFIX + ".foo.assertingparty.singlesignon.url=https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/SSOService.php",
377376
PREFIX + ".foo.assertingparty.singlesignon.binding=post",
378377
PREFIX + ".foo.assertingparty.singlesignon.sign-request=false",
379378
PREFIX + ".foo.assertingparty.entity-id=https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/metadata.php",
380-
PREFIX + ".foo.assertingparty.verification.credentials[0].bundle.name=saml",
381-
PREFIX + ".foo.assertingparty.verification.credentials[0].bundle.alias=key-alias",
379+
PREFIX + ".foo.assertingparty.verification.credentials[0].bundle=saml",
382380
PREFIX + ".foo.asserting-party.singlelogout.url=https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/SLOService.php",
383381
PREFIX + ".foo.asserting-party.singlelogout.response-url=https://simplesaml-for-spring-saml.cfapps.io/",
384382
PREFIX + ".foo.asserting-party.singlelogout.binding=post",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,147 @@
1+
/*
2+
* Copyright 2012-2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package org.springframework.boot.ssl;
18+
19+
import java.security.GeneralSecurityException;
20+
import java.security.Key;
21+
import java.security.KeyStore;
22+
import java.security.KeyStoreException;
23+
import java.security.PrivateKey;
24+
import java.security.UnrecoverableKeyException;
25+
import java.security.cert.Certificate;
26+
import java.security.cert.X509Certificate;
27+
import java.util.Enumeration;
28+
29+
import org.springframework.util.Assert;
30+
31+
/**
32+
* Provides access to private keys and certificates from a {@link KeyStore} created from
33+
* an {@link SslBundle}.
34+
*
35+
* @author Scott Frederick
36+
* @since 3.4.0
37+
*/
38+
public final class SslBundleKeyStore {
39+
40+
private final KeyStore keyStore;
41+
42+
private final SslBundleKey sslBundleKey;
43+
44+
private SslBundleKeyStore(KeyStore keyStore, SslBundleKey sslBundleKey) {
45+
this.sslBundleKey = sslBundleKey;
46+
this.keyStore = keyStore;
47+
}
48+
49+
/**
50+
* Get the private key with the alias provided by {@link SslBundleKey#getAlias()} if
51+
* the alias is specified; otherwise get the first {@link PrivateKey} in the
52+
* {@link KeyStore} that can be retrieved using {@link SslBundleKey#getPassword()}.
53+
* @return the private key, or {@code null} if no alias is specified and there is no
54+
* {@code PrivateKey} in the {@code KeyStore}
55+
*/
56+
public PrivateKey getPrivateKey() {
57+
String keyAlias = this.sslBundleKey.getAlias();
58+
String keyPassword = this.sslBundleKey.getPassword();
59+
try {
60+
if (keyAlias != null) {
61+
Key key = this.keyStore.getKey(keyAlias, keyPassword.toCharArray());
62+
Assert.notNull(key, "Private key with alias '" + keyAlias + "' was not found in SSL bundle");
63+
Assert.isInstanceOf(PrivateKey.class, key,
64+
"Key with alias '" + keyAlias + "' was expected to be a PrivateKey");
65+
return (PrivateKey) key;
66+
}
67+
return getFirstPrivateKey(this.keyStore, keyPassword);
68+
}
69+
catch (UnrecoverableKeyException kex) {
70+
throw new IllegalArgumentException("Key with alias '" + keyAlias
71+
+ "' could not be retrieved from the key store with the provided password", kex);
72+
}
73+
catch (GeneralSecurityException ex) {
74+
throw new IllegalStateException("Error getting private key from SSL bundle", ex);
75+
}
76+
}
77+
78+
/**
79+
* Get the certificate with the alias provided by {@link SslBundleKey#getAlias()} if
80+
* the alias is specified; otherwise get the first {@link X509Certificate} in the
81+
* {@link KeyStore}.
82+
* @return the certificate, or {@code null} if no alias is specified and there is no
83+
* {@code X509Certificate} in the {@code KeyStore}
84+
*/
85+
public X509Certificate getCertificate() {
86+
String keyAlias = (this.sslBundleKey != null) ? this.sslBundleKey.getAlias() : null;
87+
try {
88+
if (keyAlias != null) {
89+
Certificate certificate = this.keyStore.getCertificate(keyAlias);
90+
Assert.notNull(certificate, "Certificate with alias '" + keyAlias + "' was not found in SSL bundle");
91+
Assert.isInstanceOf(X509Certificate.class, certificate,
92+
"Certificate with alias '" + keyAlias + "' was expected to be an X509Certificate");
93+
return (X509Certificate) certificate;
94+
}
95+
return getFirstCertificate(this.keyStore);
96+
}
97+
catch (GeneralSecurityException ex) {
98+
throw new IllegalStateException("Error getting X509 certificate from SSL bundle", ex);
99+
}
100+
}
101+
102+
private PrivateKey getFirstPrivateKey(KeyStore keyStore, String keyPassword) throws KeyStoreException {
103+
Enumeration<String> aliases = keyStore.aliases();
104+
while (aliases.hasMoreElements()) {
105+
String alias = aliases.nextElement();
106+
if (keyStore.isKeyEntry(alias)) {
107+
try {
108+
Key key = keyStore.getKey(alias, (keyPassword != null) ? keyPassword.toCharArray() : null);
109+
if (key instanceof PrivateKey privateKey) {
110+
return privateKey;
111+
}
112+
}
113+
catch (UnrecoverableKeyException kex) {
114+
// password does not match this key, keep looking
115+
}
116+
catch (GeneralSecurityException ex) {
117+
throw new IllegalStateException("Error getting private key from SSL bundle", ex);
118+
}
119+
}
120+
}
121+
return null;
122+
}
123+
124+
private X509Certificate getFirstCertificate(KeyStore keyStore) throws KeyStoreException {
125+
Enumeration<String> aliases = keyStore.aliases();
126+
while (aliases.hasMoreElements()) {
127+
String alias = aliases.nextElement();
128+
Certificate certificate = keyStore.getCertificate(alias);
129+
if (certificate instanceof X509Certificate) {
130+
return (X509Certificate) certificate;
131+
}
132+
}
133+
return null;
134+
}
135+
136+
/**
137+
* Construct a new {@link SslBundleKeyStore} from the provided {@link KeyStore} and
138+
* {@link SslBundleKey}.
139+
* @param keyStore the {@code KeyStore}
140+
* @param sslBundleKey the {@code SslBundleKey}
141+
* @return an {@link SslBundleKeyStore}
142+
*/
143+
public static SslBundleKeyStore from(KeyStore keyStore, SslBundleKey sslBundleKey) {
144+
return new SslBundleKeyStore(keyStore, sslBundleKey);
145+
}
146+
147+
}

0 commit comments

Comments
 (0)