Skip to content
Snippets Groups Projects
Commit b853d28f authored by Nicholas Blair's avatar Nicholas Blair
Browse files

Merge branch 'simulate-shib' into 'master'

feat: Add optional REST Controller to simulate Shibboleth's session handler

This pull requests adds an optional REST Controller to simulate a response that matches the exact JSON format of Shibboleth's session handler.

Example: log in to https://test.my.wisc.edu. After completing login, visit https://test.my.wisc.edu/Shibboleth.sso/Session.json.

This controller is not active by default; in order to add it, activate the Spring Profile named `edu.wisc.uwss.simulated-shibboleth`.

Sample response for Amy Administrator:

```
{
  expiration: 480,
  client_address: "0:0:0:0:0:0:0:1",
  protocol: "urn:oasis:names:tc:SAML:2.0:protocol",
  identity_provider: "https://logintest.wisc.edu/idp/shibboleth",
  authn_instant: "2016-05-16T17:40:37.762",
  authncontext_class: "urn:oasis:names:tc:SAML:2.0:ac:classes:PasswordProtectedTransport",
  attributes: [
    {
      name: "persistent-id",
      values: [ "https://logintest.wisc.edu/idp/shibboleth!https://fake.wisc.edu/shibboleth!thisis/fake/PE="
]
    },
    {
      name: "uid",
      values: [ "admin" ]
    },
    {
      name: "pubcookie-user",
      values: [ "admin" ]
    },
    {
    name: "wiscEduPVI",
    values: [ "UW000A000" ]
    }
  ]
}
```

The idea for this came from a discussion with @levett.
Also notify @timothy-vertein @andrew-petro 

See merge request !11
parents 54b95e3c 96863d54
No related branches found
No related tags found
1 merge request!11feat: Add optional REST Controller to simulate Shibboleth's session handler
......@@ -106,13 +106,13 @@
<profile>
<id>combined</id>
<properties>
<activeProfiles>local-users,preauth</activeProfiles>
<activeProfiles>local-users,preauth,edu.wisc.uwss.simulated-shibboleth</activeProfiles>
</properties>
</profile>
<profile>
<id>combined-simulate-netid</id>
<properties>
<activeProfiles>local-users,preauth,preauth-simulate-netid</activeProfiles>
<activeProfiles>local-users,preauth,preauth-simulate-netid,edu.wisc.uwss.simulated-shibboleth</activeProfiles>
</properties>
</profile>
</profiles>
......
package edu.wisc.uwss.web.shibboleth;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
/**
* Part of {@link SimulatedResponse}.
*
* @author Nicholas Blair
*/
public class Attribute {
private final String name;
private final List<String> values;
public Attribute(String name, String value) {
this(name, Arrays.asList(value));
}
public Attribute(String name, List<String> values) {
this.name = name;
this.values = Collections.unmodifiableList(values);
}
public String getName() {
return name;
}
public List<String> getValues() {
return values;
}
}
package edu.wisc.uwss.web.shibboleth;
import org.apache.commons.lang3.StringUtils;
import java.time.LocalDateTime;
import java.time.ZoneOffset;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import edu.wisc.uwss.UWUserDetails;
/**
* Object matching the JSON model from /Shibboleth.sso/Session.json.
*
* @author Nicholas Blair
*/
public class SimulatedResponse {
// computed from session length
private Integer expiration = 480;
private String client_address;
private final String protocol = "urn:oasis:names:tc:SAML:2.0:protocol";
private final String identity_provider = "https://logintest.wisc.edu/idp/shibboleth";
private final String authn_instant = LocalDateTime.now(ZoneOffset.UTC).toString();
private final String authncontext_class = "urn:oasis:names:tc:SAML:2.0:ac:classes:PasswordProtectedTransport";
private List<Attribute> attributes = new ArrayList<>();
public SimulatedResponse(UWUserDetails userDetails, String client_address) {
this.client_address = client_address;
List<Attribute> attributes = new ArrayList<>();
if(StringUtils.isNotBlank(userDetails.getEppn())) {
attributes.add(new Attribute("eppn", userDetails.getEppn()));
}
attributes.add(new Attribute("persistent-id", "https://logintest.wisc.edu/idp/shibboleth!https://fake.wisc.edu/shibboleth!thisis/fake/PE="));
attributes.add(new Attribute("uid", userDetails.getUsername()));
attributes.add(new Attribute("pubcookie-user", userDetails.getUsername()));
if(StringUtils.isNotBlank(userDetails.getPvi())) {
attributes.add(new Attribute("wiscEduPVI", userDetails.getPvi()));
}
this.attributes = Collections.unmodifiableList(attributes);
}
public Integer getExpiration() {
return expiration;
}
public String getClient_address() {
return client_address;
}
public String getProtocol() {
return protocol;
}
public String getIdentity_provider() {
return identity_provider;
}
public String getAuthn_instant() {
return authn_instant;
}
public String getAuthncontext_class() {
return authncontext_class;
}
public List<Attribute> getAttributes() {
return attributes;
}
}
package edu.wisc.uwss.web.shibboleth;
import org.springframework.context.annotation.Profile;
import org.springframework.security.core.annotation.AuthenticationPrincipal;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.ResponseBody;
import java.util.Collections;
import javax.servlet.http.HttpServletRequest;
import edu.wisc.uwss.UWUserDetails;
/**
* {@link Controller} intended to simulate the JSON response from /Shibboleth.sso/Session.json,
* using the current authenticated {@link UWUserDetails} as the principal.
*
* @author Nicholas Blair
*/
@Controller
@Profile("edu.wisc.uwss.simulated-shibboleth")
public class SimulatedShibbolethSessionController {
@RequestMapping(value="/Shibboleth.sso/Session.json", method=RequestMethod.GET)
public @ResponseBody Object shibbolethSessionProfile(@AuthenticationPrincipal Object principal, HttpServletRequest request) {
if(principal instanceof UWUserDetails) {
return new SimulatedResponse((UWUserDetails) principal, request.getRemoteAddr());
} else {
return Collections.emptyMap();
}
}
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment