blob: 3953da8354a332797f90d6673c3881c361e5ec6f [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache 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://www.apache.org/licenses/LICENSE-2.0
*
* 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.apache.sling.jcr.jackrabbit.base.security;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import javax.jcr.RepositoryException;
import org.apache.jackrabbit.core.security.principal.PrincipalProvider;
import org.apache.jackrabbit.core.security.principal.PrincipalProviderRegistry;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceReference;
import org.osgi.util.tracker.ServiceTracker;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class PrincipalProviderTracker extends ServiceTracker implements PrincipalProviderRegistry{
/**
* Property-Key if the <code>PrincipalProvider</code> configured with
* {@link LoginModuleConfig#PARAM_PRINCIPAL_PROVIDER_CLASS} be registered using the
* specified name instead of the class name which is used by default if the
* name parameter is missing.
*
* @see <a href="https://issues.apache.org/jira/browse/JCR-2629">JCR-2629</a>
*/
private static final String COMPAT_PRINCIPAL_PROVIDER_NAME = "principal_provider.name";
public static final PrincipalProvider[] EMPTY_ARRAY = new PrincipalProvider[0];
private Logger log = LoggerFactory.getLogger(getClass());
private final Map<String, PrincipalProvider> providers = new ConcurrentHashMap<String, PrincipalProvider>();
private final Map<ServiceReference, String> refToNameMapping = new ConcurrentHashMap<ServiceReference, String>();
private PrincipalProvider[] providerArray = EMPTY_ARRAY;
public PrincipalProviderTracker(BundleContext context) {
super(context, PrincipalProvider.class.getName(), null);
}
//~-------------------------------------< ServiceTracker >
@Override
public Object addingService(ServiceReference reference) {
PrincipalProvider provider = (PrincipalProvider) super.addingService(reference);
addProvider(provider,reference);
reloadProviders();
return provider;
}
@Override
public void modifiedService(ServiceReference reference, Object service) {
//Check if the name has changed then re-register the provider
String newName = getProviderName((PrincipalProvider) service,reference);
String oldName = refToNameMapping.get(reference);
if(!equals(newName,oldName)){
if(oldName != null){
providers.remove(oldName);
}
addProvider((PrincipalProvider) service, reference);
reloadProviders();
}
}
@Override
public void removedService(ServiceReference reference, Object service) {
String name = refToNameMapping.remove(reference);
if(name != null){
providers.remove(name);
}
reloadProviders();
}
@Override
public void close() {
super.close();
providers.clear();
refToNameMapping.clear();
providerArray = EMPTY_ARRAY;
}
//~-------------------------------------< PrincipalProviderRegistry >
public PrincipalProvider registerProvider(Properties configuration) throws RepositoryException {
throw new UnsupportedOperationException("The PrincipalProvider are only registered as OSGi services");
}
public PrincipalProvider getDefault() {
throw new UnsupportedOperationException("Default provider is handled via WorkspaceBasedPrincipalProviderRegistry");
}
public PrincipalProvider getProvider(String className) {
return providers.get(className);
}
public PrincipalProvider[] getProviders() {
return providerArray;
}
private void addProvider(PrincipalProvider provider,ServiceReference reference){
String providerName = getProviderName(provider,reference);
if(providers.containsKey(providerName)){
log.warn("Provider with name {} is already registered. PrincipalProvider {} " +
"would not be registered",providerName,reference);
return;
}
providers.put(providerName,provider);
refToNameMapping.put(reference,providerName);
}
private void reloadProviders() {
PrincipalProvider[] providerArray = providers.values().toArray(new PrincipalProvider[providers.size()]);
synchronized (this){
this.providerArray = providerArray;
}
}
private static String getProviderName(PrincipalProvider provider,ServiceReference ref){
String providerName = (String) ref.getProperty(COMPAT_PRINCIPAL_PROVIDER_NAME);
if(providerName == null){
providerName = provider.getClass().getName();
}
return providerName;
}
private static boolean equals(Object object1, Object object2) {
if (object1 == object2) {
return true;
}
if ((object1 == null) || (object2 == null)) {
return false;
}
return object1.equals(object2);
}
}