Authentication and Authorization Infrastructure (AAI) Configuration
This page describes how to configure Opencast to take advantage of the Authentication and Authorization Infrastructure (AAI).
Prerequisites
This guides assumes that you know how to setup and configure a Shibboleth Service Provider, i.e. you are assumed to already have performed the following steps:
- Registration of your Shibboleth Service Provider at your Shibboleth Federation Service Registry
- Setup and configuration of Shibboleth on the servers you want to use it
- Configuration of your web server
In case you require help on this, contact the institution responsible for managing the Shibboleth Federation you are part of.
An informative list of Shibboleth Federations can be found on:
https://refeds.org/federations
Step 1: Configuration of the AAI Login handler
Opencast ships with a configurable AAI Login handler that needs to be adjusted to your environment.
The configuration can be found in etc/org.opencastproject.security.aai.ConfigurableLoginHandler.cfg
.
First off all, enable the AAI login handler:
enabled=true
For bootstrapping purposes, you might want to configure the AAI bootstrap user:
bootstrap.user.id=<AAI ID>
That user will be assigned ROLE_ADMIN at login time. This enables you to access the administrative UI and configure user authorization without the need to fiddle with the database directly. Once user authorization has been setup, disable the AAI bootstrap user.
Since the HTTP request header names required by the AAI login handler are specific to Shibboleth Federations, you will need to first adjust the following properties.
Set the following header names to the correct values:
header.given_name = "<Name of Shibboleth attribute>"
header.surname = "<Name of Shibboleth attribute>"
header.email = "<Name of Shibboleth attribute>"
header.home_organization = "<Name of Shibboleth attribute>"
Optionally, you can configure the name of some basic roles the AAI login handler will assign to authenticated users.
The prefix of the user role will determine what unique role a given Shibboleth user has. The role is of the form role.user.prefix + Unique ID provided by Shibboleth.
role.user.prefix = "ROLE_AAI_USER_"
To indicate the AAI home organization a user belongs to, the organization membership role is assigned to the user. The role is of the form role.organization.prefix + Home Organization provided by Shibboleth + role.organization.suffix
role.organization.prefix = "ROLE_AAI_ORG_"
role.organization.suffix = "_MEMBER"
To indicate the fact that a user has authenticated himself using Shibboleth, the login handler assigns the role as specified by the property role.federation.
role.federation = "ROLE_AAI_USER"
Step 2: Spring Security Configuration
In order to take advantage of Shibboleth authentication, you will need to uncomment the following lines found
in etc/security/mh_default_org.xml
:
The Shibboleth header authentication filter needs to be enabled to get access to the Shibboleth information within the HTTP request headers.
<!-- Shibboleth header authentication filter -->
<sec:custom-filter ref="shibbolethHeaderFilter" position="PRE_AUTH_FILTER"/>
To ensure that a logout is not just logging out the user from the Opencast application but also from Shibboleth, you will need to configure the defaultTargetUrl inside the logoutSuccessHandler:
<bean id="logoutSuccessHandler" class="org.opencastproject.kernel.security.LogoutSuccessHandler">
<property name="userDirectoryService" ref="userDirectoryService" />
<!-- Shibboleth log out -->
<property name="defaultTargetUrl" value="/Shibboleth.sso/Logout?return=www.opencast.org"/>
</bean>
IMPORTANT: In the section Shibboleth Support, be sure to adapt the value of principalRequestHeader to the respective name of the Shibboleth attribute you use in your Shibboleth Federation:
<!-- ###################### -->
<!-- # Shibboleth Support # -->
<!-- ###################### -->
<!-- General Shibboleth header extraction filter -->
<bean id="shibbolethHeaderFilter"
class="org.opencastproject.security.shibboleth.ShibbolethRequestHeaderAuthenticationFilter">
<property name="principalRequestHeader" value="<Shibboleth attribute name>"/>
<property name="authenticationManager" ref="authenticationManager" />
<property name="userDetailsService" ref="userDetailsService" />
<property name="userDirectoryService" ref="userDirectoryService" />
<property name="shibbolethLoginHandler" ref="configurableLoginHandler" />
<property name="exceptionIfHeaderMissing" value="false" />
</bean>
<!-- AAI specific header extractor and user generator -->
<bean id="configurableLoginHandler" class="org.opencastproject.security.aai.ConfigurableLoginHandler">
<property name="securityService" ref="securityService" />
<property name="userReferenceProvider" ref="userReferenceProvider" />
</bean>
<bean id="preauthAuthProvider"
class="org.springframework.security.web.authentication.preauth.PreAuthenticatedAuthenticationProvider">
<property name="preAuthenticatedUserDetailsService">
<bean id="userDetailsServiceWrapper"
class="org.springframework.security.core.userdetails.UserDetailsByNameServiceWrapper">
<property name="userDetailsService" ref="userDetailsService"/>
</bean>
</property>
</bean>
Finally be sure to enable the user reference provider to enable support for externally provided users:
<osgi:reference id="userReferenceProvider" cardinality="1..1"
interface="org.opencastproject.userdirectory.api.UserReferenceProvider" />
Since the Opencast login page is not used when Shibboleth authentication is in place, there is no point in redirecting unauthenticated requests to the Opencast login form. You can redirect them directly to the administrative user interface which is supposed to be protected by Shibboleth.
<!-- Redirects unauthenticated requests to the login form -->
<bean id="userEntryPoint" class="org.springframework.security.web.authentication.LoginUrlAuthenticationEntryPoint">
<property name="loginFormUrl" value="/admin-ng/index.html" />
</bean>
Last but not least, you need to add the preauthAuthProvider authentication provider to the authentication-manager:
<sec:authentication-manager alias="authenticationManager">
<sec:authentication-provider ref="preauthAuthProvider">
<sec:authentication-provider user-service-ref="userDetailsService">
<sec:password-encoder hash="md5"><sec:salt-source user-property="username" /></sec:password-encoder>
</sec:authentication-provider>
</sec:authentication-manager>
Step 3: Protecting HTML pages by Shibboleth
It is important to understand that Shibboleth is only used to protect content that is accessed by human users. Access to APIs is protected by other means of authentication as, for example, digest authentication.
To protect HTML pages, you will need to adapt the configuration of your web server:
<LocationMatch \.(htm|html)$>
AuthType shibboleth
ShibRequireSession On
ShibUseHeaders On
require valid-user
</LocationMatch>
Step 4: Configuring engage-UI
By default the engage-UI uses a standard login form. To make use of an AAI login we need to redirect the loginbutton to the AAI login URL:
etc/ui-config/mh_default_org/engage-ui/config.yml
:
customLoginURL: /Shibboleth.sso/Login?target=/engage/ui/index.html
The logout works out of the box since the logoutbutton calls /j_spring_security_logout
which then will redirect to the AAI logout location
configured earlier in the logoutSuccessHandler.
Advanced SSO configuration: The DynamicLoginHandler
To configure complex mappings of AAI attributes via SpEL
in the mh_default_org.xml
file, it is necessary to follow the same steps as above. Just change the bean reference of shibbolethLoginHandler
to aaiLoginHandler
in the shibbolethHeaderFilter
:
<bean id="shibbolethHeaderFilter"
class="org.opencastproject.security.shibboleth.ShibbolethRequestHeaderAuthenticationFilter">
<property name="principalRequestHeader" value="<Shibboleth attribute name>"/>
<property name="authenticationManager" ref="authenticationManager" />
<property name="userDetailsService" ref="userDetailsService" />
<property name="userDirectoryService" ref="userDirectoryService" />
<property name="shibbolethLoginHandler" ref="aaiLoginHandler" />
<property name="exceptionIfHeaderMissing" value="false" />
</bean>
Activate the aaiLoginHandler
bean and the attributeMapper
bean
<bean id="aaiLoginHandler" class="org.opencastproject.security.aai.DynamicLoginHandler">
<property name="securityService" ref="securityService" />
<property name="userReferenceProvider" ref="userReferenceProvider" />
<property name="attributeMapper" ref="attributeMapper" />
</bean>
<bean id="attributeMapper" class="org.opencastproject.security.aai.api.AttributeMapper">
<property name="useHeader" value="true" />
<property name="multiValueDelimiter" value=";" />
<property name="attributeMap" ref="attributeMap" />
<property name="aaiAttributes" ref="aaiAttributes" />
</bean>
then, define all the attributes you may want to use, so the mapper gets populated at login e.g.
<util:list id="aaiAttributes" value-type="java.lang.String">
<value>sn</value>
<value>givenName</value>
<value>mail</value>
<value>homeOrganization</value>
<value>eduPersonEntitlement</value>
<value>eduPersonPrincipalName</value>
<value>homeOrganization</value>
</util:list>
Opencast has a fairly simple account model that consists only of an username, display name and email address. Additionally an user may have some roles. For each of those attributes you define a map entry that refers to a list of mappings (username is mapped in the shibbolethHeaderFilter
bean above).
<util:map id="attributeMap" map-class="java.util.HashMap">
<entry key="roles" value-ref="roleMapping" />
<entry key="displayName" value-ref="displayNameMapping" />
<entry key="mail" value-ref="mailMapping" />
</util:map>
Each attribute as treated as multi-value so you have to access single value attributes (like sn
and givenName
) by accessing the first element of the list of values received via Shibboleth. For the email address and the displayName
of an user the mapping is usually straight forward, see inline comments for further explanation:
<!-- Use SpEL string concatenation to build a displayName from `sn` and `givenName` -->
<util:list id="displayNameMapping" value-type="java.lang.String">
<value>['givenName'][0] + ' ' + ['sn'][0]</value>
</util:list>
<util:list id="mailMapping" value-type="java.lang.String">
<value>['mail'][0]</value>
</util:list>
The mail
and the displayName
of an user has only one value, but when it comes to the roles of an user you are able to produce an arbitrary number of roles using the whole power of SpEL and java.lang.String
methods. See inline comments for further explanation and examples:
<util:list id="roleMapping" value-type="java.lang.String">
<!-- assign every user the role ROLE_AAI_USER -->
<value>'ROLE_AAI_USER'</value>
<!-- assign every user the role ROLE_AAI_USER_<username>.
For a user john.doe@example.org this will result in a ROLE_USER_JOHN_DOE_EXAMPLE_ORG
since Opencast replaces all special chars in this case -->
<value>'ROLE_AAI_USER_' + ['eduPersonPrincipalName']</value>
<!-- if you implement a kind of DAC based access control you may want to produce
an owner role. To get the same behaviour as for ROLE_AAI_USER_<username> above
you may replace all special chars from the value of `eduPersonPrincipalName`
and convert it to uppercase -->
<value>('ROLE_AAI_OWNER_' + ['eduPersonPrincipalName']).replaceAll("[^a-zA-Z0-9]","_").toUpperCase()</value>
<!-- an AAI user may provide the attribute homeOrganization.
If so, assign an appropriate role. Otherwise not. -->
<value>['homeOrganization'] != null ? 'ROLE_AAI_ORG_' + ['homeOrganization'] + '_MEMBER' : null</value>
<!-- a sophisticated AAI approach would use entitlements to assign
e.g. the admin role to authorized users -->
<value>['eduPersonEntitlement'].contains('urn:mace:opencast.org:permission:shibboleth:opencast_admin') ? 'ROLE_ADMIN' : null</value>
<!-- a less sophisticated approach would directly assign the admin role
to users based on usernames (same as bootstrap user above) -->
<value>['eduPersonPrincipalName'].contains('john.doe@example.org') ? 'ROLE_ADMIN' : null</value>
<!-- to assing a common set of roles to a class of users, you may create
a group (e.g. AAI_EDITOR) and assign some roles to it in the admin ui.
So the group is the container of all the roles and you assign only the
group to a user e.g. based on his affiliation, thus enabling a class
of users for certain abilities in Opencast -->
<value>['eduPersonScopedAffiliation'].contains('faculty@example.org') ? 'ROLE_GROUP_AAI_EDITOR' : null</value>
</util:list>
You may start by uncommenting the whole example in mh_default_org.xml
.