BrightspaceUserProviderFactory.java
/*
* Licensed to The Apereo Foundation under one or more contributor license
* agreements. See the NOTICE file distributed with this work for additional
* information regarding copyright ownership.
*
*
* The Apereo Foundation licenses this file to you under the Educational
* Community License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of the License
* at:
*
* http://opensource.org/licenses/ecl2.txt
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*
*/
package org.opencastproject.userdirectory.brightspace;
import org.opencastproject.security.api.Organization;
import org.opencastproject.security.api.OrganizationDirectoryService;
import org.opencastproject.security.api.RoleProvider;
import org.opencastproject.security.api.SecurityConstants;
import org.opencastproject.security.api.UserProvider;
import org.opencastproject.userdirectory.brightspace.client.BrightspaceClientImpl;
import org.opencastproject.util.NotFoundException;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.math.NumberUtils;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.osgi.service.cm.ConfigurationException;
import org.osgi.service.cm.ManagedServiceFactory;
import org.osgi.service.component.ComponentContext;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.lang.management.ManagementFactory;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Dictionary;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
@Component(
immediate = true,
service = ManagedServiceFactory.class,
property = {
"service.pid=org.opencastproject.userdirectory.brightspace",
"service.description=Provides Brightspace user directory instances"
}
)
public class BrightspaceUserProviderFactory implements ManagedServiceFactory {
private static final Logger logger = LoggerFactory.getLogger(BrightspaceUserProviderFactory.class);
private static final String LTI_LEARNER_ROLE = "Learner";
private static final String LTI_INSTRUCTOR_ROLE = "Instructor";
private static final String ORGANIZATION_KEY = "org.opencastproject.userdirectory.brightspace.org";
private static final String BRIGHTSPACE_USER_ID = "org.opencastproject.userdirectory.brightspace.systemuser.id";
private static final String BRIGHTSPACE_USER_KEY = "org.opencastproject.userdirectory.brightspace.systemuser.key";
private static final String BRIGHTSPACE_URL = "org.opencastproject.userdirectory.brightspace.url";
private static final String BRIGHTSPACE_APP_ID = "org.opencastproject.userdirectory.brightspace.application.id";
private static final String BRIGHTSPACE_APP_KEY = "org.opencastproject.userdirectory.brightspace.application.key";
private static final String CACHE_SIZE_KEY = "org.opencastproject.userdirectory.brightspace.cache.size";
private static final String CACHE_EXPIRATION_KEY = "org.opencastproject.userdirectory.brightspace.cache.expiration";
private static final String BRIGHTSPACE_NAME = "org.opencastproject.userdirectory.brightspace";
private static final int DEFAULT_CACHE_SIZE_VALUE = 1000;
private static final int DEFAULT_CACHE_EXPIRATION_VALUE = 60;
/** The keys to look up which roles in Brightspace should be considered as instructor roles */
private static final String BRIGHTSPACE_INSTRUCTOR_ROLES_KEY =
"org.opencastproject.userdirectory.brightspace.instructor.roles";
private static final String DEFAULT_BRIGHTSPACE_INSTRUCTOR_ROLES = "teacher,ta";
/** The keys to look up which users should be ignored */
private static final String IGNORED_USERNAMES_KEY = "org.opencastproject.userdirectory.brightspace.ignored.usernames";
private static final String DEFAULT_IGNORED_USERNAMES = "admin,anonymous";
protected BundleContext bundleContext;
private Map<String, ServiceRegistration> providerRegistrations = new ConcurrentHashMap<>();
private OrganizationDirectoryService orgDirectory;
private int cacheSize;
private int cacheExpiration;
/**
* Builds a JMX object name for a given PID
*
* @param pid the PID
* @return the object name
* @throws NullPointerException
* @throws MalformedObjectNameException
*/
public static final ObjectName getObjectName(String pid) throws MalformedObjectNameException, NullPointerException {
return new ObjectName(pid + ":type=BrightspaceRequests");
}
/**
* OSGi callback for setting the organization directory service.
*/
@Reference
public void setOrgDirectory(OrganizationDirectoryService orgDirectory) {
this.orgDirectory = orgDirectory;
}
/**
* Callback for the activation of this component
*
* @param cc the component context
*/
@Activate
public void activate(ComponentContext cc) {
logger.debug("Activate BrightspaceUserProviderFactory");
this.bundleContext = cc.getBundleContext();
}
/**
* {@inheritDoc}
*
* @see org.osgi.service.cm.ManagedServiceFactory#getName()
*/
@Override
public String getName() {
return BRIGHTSPACE_NAME;
}
/**
* {@inheritDoc}
*
* @see org.osgi.service.cm.ManagedServiceFactory#updated(java.lang.String, java.util.Dictionary)
*/
@Override
public void updated(String pid, Dictionary properties) throws ConfigurationException {
logger.debug("updated BrightspaceUserProviderFactory");
String adminUserName = StringUtils.trimToNull(
bundleContext.getProperty(SecurityConstants.GLOBAL_ADMIN_USER_PROPERTY));
String organization = (String) properties.get(ORGANIZATION_KEY);
String urlStr = (String) properties.get(BRIGHTSPACE_URL);
String systemUserId = (String) properties.get(BRIGHTSPACE_USER_ID);
String systemUserKey = (String) properties.get(BRIGHTSPACE_USER_KEY);
final String applicationId = (String) properties.get(BRIGHTSPACE_APP_ID);
final String applicationKey = (String) properties.get(BRIGHTSPACE_APP_KEY);
String cacheSizeStr = (String) properties.get(CACHE_SIZE_KEY);
if (StringUtils.isBlank(cacheSizeStr)) {
cacheSize = DEFAULT_CACHE_SIZE_VALUE;
} else {
cacheSize = NumberUtils.toInt(cacheSizeStr);
}
String cacheExpirationStr = (String) properties.get(CACHE_EXPIRATION_KEY);
if (StringUtils.isBlank(cacheExpirationStr)) {
cacheExpiration = DEFAULT_CACHE_EXPIRATION_VALUE;
} else {
cacheExpiration = NumberUtils.toInt(cacheExpirationStr);
}
String rolesStr = (String) properties.get(BRIGHTSPACE_INSTRUCTOR_ROLES_KEY);
if (StringUtils.isBlank(rolesStr)) {
rolesStr = DEFAULT_BRIGHTSPACE_INSTRUCTOR_ROLES;
}
Set instructorRoles = parsePropertyLineAsSet(rolesStr);
logger.debug("Brightspace instructor roles: {}", instructorRoles);
String ignoredUsersStr = (String) properties.get(IGNORED_USERNAMES_KEY);
if (StringUtils.isBlank(ignoredUsersStr)) {
ignoredUsersStr = DEFAULT_IGNORED_USERNAMES;
}
Set ignoredUsernames = parsePropertyLineAsSet(ignoredUsersStr);
logger.debug("Ignored users: {}", ignoredUsernames);
validateUrl(urlStr);
validateConfigurationKey(ORGANIZATION_KEY, organization);
validateConfigurationKey(BRIGHTSPACE_USER_ID, systemUserId);
validateConfigurationKey(BRIGHTSPACE_USER_KEY, systemUserKey);
validateConfigurationKey(BRIGHTSPACE_APP_ID, applicationId);
validateConfigurationKey(BRIGHTSPACE_APP_KEY, applicationKey);
ServiceRegistration existingRegistration = this.providerRegistrations.remove(pid);
if (existingRegistration != null) {
existingRegistration.unregister();
}
Organization org;
try {
org = this.orgDirectory.getOrganization(organization);
} catch (NotFoundException nfe) {
logger.warn("Organization {} not found!", organization);
throw new ConfigurationException(ORGANIZATION_KEY, "not found");
}
logger.debug("creating new brightspace user provider for pid={}", pid);
BrightspaceClientImpl clientImpl
= new BrightspaceClientImpl(urlStr, applicationId, applicationKey, systemUserId, systemUserKey);
BrightspaceUserProviderInstance provider
= new BrightspaceUserProviderInstance(pid, clientImpl, org, cacheSize, cacheExpiration ,
instructorRoles, ignoredUsernames);
this.providerRegistrations
.put(pid, this.bundleContext.registerService(UserProvider.class.getName(), provider, null));
this.providerRegistrations
.put(pid, this.bundleContext.registerService(RoleProvider.class.getName(), provider, null));
}
/**
* {@inheritDoc}
*
* @see org.osgi.service.cm.ManagedServiceFactory#deleted(java.lang.String)
*/
@Override
public void deleted(String pid) {
logger.debug("delete BrightspaceUserProviderInstance for pid={}", pid);
ServiceRegistration registration = providerRegistrations.remove(pid);
if (registration != null) {
registration.unregister();
try {
ManagementFactory.getPlatformMBeanServer().unregisterMBean(getObjectName(pid));
} catch (Exception e) {
logger.warn("Unable to unregister mbean for pid='{}'", pid, e);
}
}
}
private void validateConfigurationKey(String key, String value) throws ConfigurationException {
if (StringUtils.isBlank(value)) {
throw new ConfigurationException(key, "is not set");
}
}
private void validateUrl(String urlStr) throws ConfigurationException {
if (StringUtils.isBlank(urlStr)) {
throw new ConfigurationException(BRIGHTSPACE_URL, "is not set");
} else {
try {
new URI(urlStr);
} catch (URISyntaxException e) {
throw new ConfigurationException(BRIGHTSPACE_URL, "not a URL");
}
}
}
private Set<String> parsePropertyLineAsSet(String configLine) {
Set<String> set = new HashSet<>();
String[] configs = configLine.split(",");
for (String config: configs) {
set.add(config.trim());
}
return set;
}
}