/* * Copyright 2016 Red Hat, Inc. and/or its affiliates * and other contributors as indicated by the @author tags. * * Licensed 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.keycloak.storage.ldap; import org.jboss.logging.Logger; import org.keycloak.common.util.MultivaluedHashMap; import org.keycloak.component.ComponentModel; import org.keycloak.models.KeycloakSession; import org.keycloak.models.RealmModel; import org.keycloak.storage.ldap.idm.store.ldap.LDAPIdentityStore; import org.keycloak.storage.ldap.mappers.LDAPConfigDecorator; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.stream.Collectors; /** * @author <a href="mailto:mposolda@redhat.com">Marek Posolda</a> */ public class LDAPIdentityStoreRegistry { private static final Logger logger = Logger.getLogger(LDAPIdentityStoreRegistry.class); private Map<String, LDAPIdentityStoreContext> ldapStores = new ConcurrentHashMap<>(); public LDAPIdentityStore getLdapStore(KeycloakSession session, ComponentModel ldapModel, Map<ComponentModel, LDAPConfigDecorator> configDecorators) { LDAPIdentityStoreContext context = ldapStores.get(ldapModel.getId()); // Ldap config might have changed for the realm. In this case, we must re-initialize MultivaluedHashMap<String, String> configModel = ldapModel.getConfig(); LDAPConfig ldapConfig = new LDAPConfig(configModel); for (Map.Entry<ComponentModel, LDAPConfigDecorator> entry : configDecorators.entrySet()) { ComponentModel mapperModel = entry.getKey(); LDAPConfigDecorator decorator = entry.getValue(); decorator.updateLDAPConfig(ldapConfig, mapperModel); } if (context == null || !ldapConfig.equals(context.config)) { logLDAPConfig(session, ldapModel, ldapConfig); LDAPIdentityStore store = createLdapIdentityStore(ldapConfig); context = new LDAPIdentityStoreContext(ldapConfig, store); ldapStores.put(ldapModel.getId(), context); } return context.store; } // Don't log LDAP password private void logLDAPConfig(KeycloakSession session, ComponentModel ldapModel, LDAPConfig ldapConfig) { logger.infof("Creating new LDAP Store for the LDAP storage provider: '%s', LDAP Configuration: %s", ldapModel.getName(), ldapConfig.toString()); if (logger.isDebugEnabled()) { RealmModel realm = session.realms().getRealm(ldapModel.getParentId()); List<ComponentModel> mappers = realm.getComponents(ldapModel.getId()); mappers.stream().forEach((ComponentModel c) -> { logger.debugf("Mapper for provider: %s, Mapper name: %s, Provider: %s, Mapper configuration: %s", ldapModel.getName(), c.getName(), c.getProviderId(), c.getConfig().toString()); }); } } /** * Create LDAPIdentityStore to be cached in the local registry */ public static LDAPIdentityStore createLdapIdentityStore(LDAPConfig cfg) { checkSystemProperty("com.sun.jndi.ldap.connect.pool.authentication", "none simple"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.initsize", "1"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.maxsize", "1000"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.prefsize", "5"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.timeout", "300000"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.protocol", "plain"); checkSystemProperty("com.sun.jndi.ldap.connect.pool.debug", "off"); return new LDAPIdentityStore(cfg); } private static void checkSystemProperty(String name, String defaultValue) { if (System.getProperty(name) == null) { System.setProperty(name, defaultValue); } } private class LDAPIdentityStoreContext { private LDAPIdentityStoreContext(LDAPConfig config, LDAPIdentityStore store) { this.config = config; this.store = store; } private LDAPConfig config; private LDAPIdentityStore store; } }