|
| 1 | +/* |
| 2 | + * Copyright 2002-2020 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 sample; |
| 18 | + |
| 19 | +import java.util.Arrays; |
| 20 | +import java.util.LinkedHashMap; |
| 21 | +import java.util.List; |
| 22 | +import java.util.Map; |
| 23 | +import javax.servlet.http.HttpSession; |
| 24 | + |
| 25 | +import com.gargoylesoftware.htmlunit.WebClient; |
| 26 | +import com.gargoylesoftware.htmlunit.html.HtmlForm; |
| 27 | +import com.gargoylesoftware.htmlunit.html.HtmlInput; |
| 28 | +import com.gargoylesoftware.htmlunit.html.HtmlPage; |
| 29 | +import com.gargoylesoftware.htmlunit.html.HtmlSubmitInput; |
| 30 | +import org.junit.Test; |
| 31 | +import org.junit.runner.RunWith; |
| 32 | + |
| 33 | +import org.springframework.beans.factory.annotation.Autowired; |
| 34 | +import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc; |
| 35 | +import org.springframework.boot.test.context.SpringBootTest; |
| 36 | +import org.springframework.mock.web.MockHttpSession; |
| 37 | +import org.springframework.test.context.junit4.SpringRunner; |
| 38 | +import org.springframework.test.web.servlet.MockMvc; |
| 39 | + |
| 40 | +import static org.assertj.core.api.Assertions.assertThat; |
| 41 | +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; |
| 42 | +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; |
| 43 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.model; |
| 44 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.redirectedUrl; |
| 45 | +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; |
| 46 | + |
| 47 | +@RunWith(SpringRunner.class) |
| 48 | +@SpringBootTest |
| 49 | +@AutoConfigureMockMvc |
| 50 | +public class Saml2LoginApplicationITests { |
| 51 | + static final String SIGNED_RESPONSE = "<samlp:Response xmlns:samlp="urn:oasis:names:tc:SAML:2.0:protocol" xmlns:saml="urn:oasis:names:tc:SAML:2.0:assertion" ID="_8e5655ed7ca55e286b6150088ec755f660f5741cc7" Version="2.0" IssueInstant="2020-08-28T21:10:25Z" Destination="http://localhost:8080/login/saml2/sso/one" InResponseTo="ARQb8f3343-d333-41a3-a194-1617aa64b3b3"><saml:Issuer>https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/metadata.php</saml:Issuer><ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
  <ds:SignedInfo><ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
    <ds:SignatureMethod Algorithm="http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"/>
  <ds:Reference URI="#_8e5655ed7ca55e286b6150088ec755f660f5741cc7"><ds:Transforms><ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/><ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/></ds:Transforms><ds:DigestMethod Algorithm="http://www.w3.org/2001/04/xmlenc#sha256"/><ds:DigestValue>3NBLZhpc0Jq9P0H8UC1OFaT7hOcD14fgyceDVwtL9T0=</ds:DigestValue></ds:Reference></ds:SignedInfo><ds:SignatureValue>cuaXoaYvcfA9h9lCQlSlUVc35z+oxHdVfAU2NlCe8pXKknQVAzfCPIRQd7rY7rBW89ZSoBlwdlKR0zlxpprJuMzvrdMe9hrmMiShVqiWC5JFhCJwJXliw0DXi9TFz6G1H0QbmgTtdd2NaB3GnVDXlF/weltiACQ3C5JvhMG7PyRgPXbEkIdYlsgafM/QaorP7MB2W5MlZxBbaekN7BoDu5MfHa1OQ+vCCHyGHPlmIN7fmo47oqXSa15rQ0oTPewYLjEE2Wk/lzZ+3u8NHp+mV9RG6uyVDq2SRlVnWPIP/JWiWEV8UEMw+D6GSr6t17Q6AevTsBFdqgP7Q22Idrg0kQ==</ds:SignatureValue>
<ds:KeyInfo><ds:X509Data><ds:X509Certificate>MIIEEzCCAvugAwIBAgIJAIc1qzLrv+5nMA0GCSqGSIb3DQEBCwUAMIGfMQswCQYDVQQGEwJVUzELMAkGA1UECAwCQ08xFDASBgNVBAcMC0Nhc3RsZSBSb2NrMRwwGgYDVQQKDBNTYW1sIFRlc3RpbmcgU2VydmVyMQswCQYDVQQLDAJJVDEgMB4GA1UEAwwXc2ltcGxlc2FtbHBocC5jZmFwcHMuaW8xIDAeBgkqhkiG9w0BCQEWEWZoYW5pa0BwaXZvdGFsLmlvMB4XDTE1MDIyMzIyNDUwM1oXDTI1MDIyMjIyNDUwM1owgZ8xCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJDTzEUMBIGA1UEBwwLQ2FzdGxlIFJvY2sxHDAaBgNVBAoME1NhbWwgVGVzdGluZyBTZXJ2ZXIxCzAJBgNVBAsMAklUMSAwHgYDVQQDDBdzaW1wbGVzYW1scGhwLmNmYXBwcy5pbzEgMB4GCSqGSIb3DQEJARYRZmhhbmlrQHBpdm90YWwuaW8wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC4cn62E1xLqpN34PmbrKBbkOXFjzWgJ9b+pXuaRft6A339uuIQeoeH5qeSKRVTl32L0gdz2ZivLwZXW+cqvftVW1tvEHvzJFyxeTW3fCUeCQsebLnA2qRa07RkxTo6Nf244mWWRDodcoHEfDUSbxfTZ6IExSojSIU2RnD6WllYWFdD1GFpBJOmQB8rAc8wJIBdHFdQnX8Ttl7hZ6rtgqEYMzYVMuJ2F2r1HSU1zSAvwpdYP6rRGFRJEfdA9mm3WKfNLSc5cljz0X/TXy0vVlAV95l9qcfFzPmrkNIst9FZSwpvB49LyAVke04FQPPwLgVH4gphiJH3jvZ7I+J5lS8VAgMBAAGjUDBOMB0GA1UdDgQWBBTTyP6Cc5HlBJ5+ucVCwGc5ogKNGzAfBgNVHSMEGDAWgBTTyP6Cc5HlBJ5+ucVCwGc5ogKNGzAMBgNVHRMEBTADAQH/MA0GCSqGSIb3DQEBCwUAA4IBAQAvMS4EQeP/ipV4jOG5lO6/tYCb/iJeAduOnRhkJk0DbX329lDLZhTTL/x/w/9muCVcvLrzEp6PN+VWfw5E5FWtZN0yhGtP9R+vZnrV+oc2zGD+no1/ySFOe3EiJCO5dehxKjYEmBRv5sU/LZFKZpozKN/BMEa6CqLuxbzb7ykxVr7EVFXwltPxzE9TmL9OACNNyF5eJHWMRMllarUvkcXlh4pux4ks9e6zV9DQBy2zds9f1I3qxg0eX6JnGrXi/ZiCT+lJgVe3ZFXiejiLAiKB04sXW3ti0LW3lx13Y1YlQ4/tlpgTgfIJxKV6nyPiLoK0nywbMd+vpAirDt2Oc+hk</ds:X509Certificate></ds:X509Data></ds:KeyInfo></ds:Signature><samlp:Status><samlp:StatusCode Value="urn:oasis:names:tc:SAML:2.0:status:Success"/></samlp:Status><saml:Assertion xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xs="http://www.w3.org/2001/XMLSchema" ID="_ba729e2caa4dfa16f7d403c17506a9135f97c007f8" Version="2.0" IssueInstant="2020-08-28T21:10:25Z"><saml:Issuer>https://simplesaml-for-spring-saml.cfapps.io/saml2/idp/metadata.php</saml:Issuer><ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
  <ds:SignedInfo><ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
    <ds:SignatureMethod Algorithm="http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"/>
  <ds:Reference URI="#_ba729e2caa4dfa16f7d403c17506a9135f97c007f8"><ds:Transforms><ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/><ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/></ds:Transforms><ds:DigestMethod Algorithm="http://www.w3.org/2001/04/xmlenc#sha256"/><ds:DigestValue>DtzI6ryJ14V7EzZFiNuPyvEFbXF1nfANNYY17AbZ110=</ds:DigestValue></ds:Reference></ds:SignedInfo><ds:SignatureValue>KCffG+BIbGx/vfUuVq2Snj8wX3Hm++OdUgFadGHak8A/XOC2pTDh6BYqacO8CqHAGoozFkpVaMOi7Nw2jERuLhh7hmFW0vhOm+g8QEq3uB0edcwliKrkTmgEcdNeEmizmyABixwDlqGwS3Jp80wkCSqK+yLAl2RhZH4CI/15keW1IDcH87//i1se9Y9SYUU6chjlV1OkC1Ytytjp5yTwZqoRjQtJ0YUJzzLUNLrZGLodntk4tW3IgaU3ow8i6t6RejQxmg4sILg178/nU7to/FgmFw6Rc8NhjanDcarf7roW+UYfBzAbcKaMMhAUyHxF0tUN1KcGx5qMqMEQNu/bDQ==</ds:SignatureValue>
<ds:KeyInfo><ds:X509Data><ds:X509Certificate>MIIEEzCCAvugAwIBAgIJAIc1qzLrv+5nMA0GCSqGSIb3DQEBCwUAMIGfMQswCQYDVQQGEwJVUzELMAkGA1UECAwCQ08xFDASBgNVBAcMC0Nhc3RsZSBSb2NrMRwwGgYDVQQKDBNTYW1sIFRlc3RpbmcgU2VydmVyMQswCQYDVQQLDAJJVDEgMB4GA1UEAwwXc2ltcGxlc2FtbHBocC5jZmFwcHMuaW8xIDAeBgkqhkiG9w0BCQEWEWZoYW5pa0BwaXZvdGFsLmlvMB4XDTE1MDIyMzIyNDUwM1oXDTI1MDIyMjIyNDUwM1owgZ8xCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJDTzEUMBIGA1UEBwwLQ2FzdGxlIFJvY2sxHDAaBgNVBAoME1NhbWwgVGVzdGluZyBTZXJ2ZXIxCzAJBgNVBAsMAklUMSAwHgYDVQQDDBdzaW1wbGVzYW1scGhwLmNmYXBwcy5pbzEgMB4GCSqGSIb3DQEJARYRZmhhbmlrQHBpdm90YWwuaW8wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC4cn62E1xLqpN34PmbrKBbkOXFjzWgJ9b+pXuaRft6A339uuIQeoeH5qeSKRVTl32L0gdz2ZivLwZXW+cqvftVW1tvEHvzJFyxeTW3fCUeCQsebLnA2qRa07RkxTo6Nf244mWWRDodcoHEfDUSbxfTZ6IExSojSIU2RnD6WllYWFdD1GFpBJOmQB8rAc8wJIBdHFdQnX8Ttl7hZ6rtgqEYMzYVMuJ2F2r1HSU1zSAvwpdYP6rRGFRJEfdA9mm3WKfNLSc5cljz0X/TXy0vVlAV95l9qcfFzPmrkNIst9FZSwpvB49LyAVke04FQPPwLgVH4gphiJH3jvZ7I+J5lS8VAgMBAAGjUDBOMB0GA1UdDgQWBBTTyP6Cc5HlBJ5+ucVCwGc5ogKNGzAfBgNVHSMEGDAWgBTTyP6Cc5HlBJ5+ucVCwGc5ogKNGzAMBgNVHRMEBTADAQH/MA0GCSqGSIb3DQEBCwUAA4IBAQAvMS4EQeP/ipV4jOG5lO6/tYCb/iJeAduOnRhkJk0DbX329lDLZhTTL/x/w/9muCVcvLrzEp6PN+VWfw5E5FWtZN0yhGtP9R+vZnrV+oc2zGD+no1/ySFOe3EiJCO5dehxKjYEmBRv5sU/LZFKZpozKN/BMEa6CqLuxbzb7ykxVr7EVFXwltPxzE9TmL9OACNNyF5eJHWMRMllarUvkcXlh4pux4ks9e6zV9DQBy2zds9f1I3qxg0eX6JnGrXi/ZiCT+lJgVe3ZFXiejiLAiKB04sXW3ti0LW3lx13Y1YlQ4/tlpgTgfIJxKV6nyPiLoK0nywbMd+vpAirDt2Oc+hk</ds:X509Certificate></ds:X509Data></ds:KeyInfo></ds:Signature><saml:Subject><saml:NameID SPNameQualifier="http://localhost:8080/saml2/service-provider-metadata/one" Format="urn:oasis:names:tc:SAML:1.1:nameid-format:emailAddress">testuser@spring.security.saml</saml:NameID><saml:SubjectConfirmation Method="urn:oasis:names:tc:SAML:2.0:cm:bearer"><saml:SubjectConfirmationData NotOnOrAfter="2052-05-06T22:57:05Z" Recipient="http://localhost:8080/login/saml2/sso/one" InResponseTo="ARQb8f3343-d333-41a3-a194-1617aa64b3b3"/></saml:SubjectConfirmation></saml:Subject><saml:Conditions NotBefore="2020-08-28T21:09:55Z" NotOnOrAfter="2052-05-06T22:57:05Z"><saml:AudienceRestriction><saml:Audience>http://localhost:8080/saml2/service-provider-metadata/one</saml:Audience></saml:AudienceRestriction></saml:Conditions><saml:AuthnStatement AuthnInstant="2020-08-28T21:10:25Z" SessionNotOnOrAfter="2020-08-29T05:10:25Z" SessionIndex="_1a7a8638e86ceaa92cc58646c4231875a6c9d1cabb"><saml:AuthnContext><saml:AuthnContextClassRef>urn:oasis:names:tc:SAML:2.0:ac:classes:PasswordProtectedTransport</saml:AuthnContextClassRef></saml:AuthnContext></saml:AuthnStatement><saml:AttributeStatement><saml:Attribute Name="uid" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">testuser@spring.security.saml</saml:AttributeValue></saml:Attribute><saml:Attribute Name="eduPersonAffiliation" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">member</saml:AttributeValue><saml:AttributeValue xsi:type="xs:string">user</saml:AttributeValue></saml:Attribute><saml:Attribute Name="emailAddress" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">testuser@spring.security.saml</saml:AttributeValue></saml:Attribute></saml:AttributeStatement></saml:Assertion></samlp:Response>"; |
| 52 | + |
| 53 | + static final Map<String, List<Object>> USER_ATTRIBUTES = new LinkedHashMap<>(); |
| 54 | + |
| 55 | + static { |
| 56 | + USER_ATTRIBUTES. put( "uid", Arrays. asList( "[email protected]")); |
| 57 | + USER_ATTRIBUTES.put("eduPersonAffiliation", Arrays.asList("member", "user")); |
| 58 | + USER_ATTRIBUTES. put( "emailAddress", Arrays. asList( "[email protected]")); |
| 59 | + } |
| 60 | + |
| 61 | + @Autowired |
| 62 | + MockMvc mvc; |
| 63 | + |
| 64 | + @Autowired |
| 65 | + WebClient webClient; |
| 66 | + |
| 67 | + @Test |
| 68 | + public void indexWhenSamlResponseThenShowsUserInformation() throws Exception { |
| 69 | + HttpSession session = this.mvc.perform(get("http://localhost:8080/")) |
| 70 | + .andExpect(status().is3xxRedirection()) |
| 71 | + .andExpect(redirectedUrl("http://localhost:8080/saml2/authenticate/one")) |
| 72 | + .andReturn() |
| 73 | + .getRequest().getSession(); |
| 74 | + |
| 75 | + this.mvc.perform(post("http://localhost:8080/login/saml2/sso/one") |
| 76 | + .param("SAMLResponse", SIGNED_RESPONSE) |
| 77 | + .session((MockHttpSession) session)) |
| 78 | + .andExpect(redirectedUrl("http://localhost:8080/")); |
| 79 | + |
| 80 | + this.mvc.perform(get("http://localhost:8080/") |
| 81 | + .session((MockHttpSession) session)) |
| 82 | + . andExpect( model(). attribute( "emailAddress", "[email protected]")) |
| 83 | + .andExpect(model().attribute("userAttributes", USER_ATTRIBUTES)); |
| 84 | + } |
| 85 | + |
| 86 | + @Test |
| 87 | + public void authenticationAttemptWhenValidThenShowsUserEmailAddress() throws Exception { |
| 88 | + HtmlPage assertingParty = this.webClient.getPage("/"); |
| 89 | + HtmlForm form = assertingParty.getFormByName("f"); |
| 90 | + HtmlInput username = form.getInputByName("username"); |
| 91 | + HtmlInput password = form.getInputByName("password"); |
| 92 | + HtmlSubmitInput submit = assertingParty.getHtmlElementById("submit_button"); |
| 93 | + username.setValueAttribute("user"); |
| 94 | + password.setValueAttribute("password"); |
| 95 | + HtmlPage relyingParty = submit.click(); |
| 96 | + assertThat(relyingParty.asText()) |
| 97 | + . contains( "You're email address is [email protected]"); |
| 98 | + } |
| 99 | +} |
0 commit comments