001/* 002 * Copyright 2015 Anyware Services 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package org.ametys.cms.search.solr; 017 018import java.util.HashMap; 019import java.util.Map; 020import java.util.Optional; 021import java.util.concurrent.TimeUnit; 022 023import javax.jcr.RepositoryException; 024 025import org.apache.avalon.framework.activity.Disposable; 026import org.apache.avalon.framework.activity.Initializable; 027import org.apache.avalon.framework.component.Component; 028import org.apache.avalon.framework.service.ServiceException; 029import org.apache.avalon.framework.service.ServiceManager; 030import org.apache.avalon.framework.service.Serviceable; 031import org.apache.commons.io.IOUtils; 032import org.apache.commons.lang3.ArrayUtils; 033import org.apache.solr.client.solrj.SolrClient; 034import org.apache.solr.client.solrj.impl.Http2SolrClient; 035 036import org.ametys.plugins.repository.provider.AbstractRepository; 037import org.ametys.plugins.repository.provider.JackrabbitRepository; 038import org.ametys.plugins.repository.provider.WorkspaceSelector; 039import org.ametys.runtime.config.Config; 040import org.ametys.runtime.plugin.component.AbstractLogEnabled; 041 042/** 043 * Component acting as a single entry point to get access to Solr clients. 044 */ 045public class DefaultSolrClientProvider extends AbstractLogEnabled implements SolrClientProvider, Component, Serviceable, Initializable, Disposable 046{ 047 private static final String __SOLR_URL_CONFIG = "cms.solr.core.url"; 048 private static final String __SOLR_SOCKET_TIMEOUT_CONFIG = "cms.solr.socket.timeout"; 049 private static final String __SOLR_CORE_PREFIX_CONFIG = "cms.solr.core.prefix"; 050 051 /** The workspace selector. */ 052 protected WorkspaceSelector _workspaceSelector; 053 054 /** The JCR repository */ 055 protected JackrabbitRepository _repository; 056 057 /** The Solr read client. */ 058 protected SolrClient _solrReadClient; 059 060 /** The Solr "default" update clients, per workspace */ 061 protected Map<String, SolrClient> _solrDefaultUpdateClients; 062 063 /** The Solr "no auto commit" update clients, per workspace */ 064 protected Map<String, SolrClient> _solrNoAutoCommitUpdateClients; 065 066 /** The solr URL. */ 067 protected String _solrUrl; 068 069 /** The solr socket timeout (in millis). */ 070 protected Optional<Integer> _solrSocketTimeout; 071 072 /** The solr core prefix. */ 073 protected String _solrCorePrefix; 074 075 @Override 076 public void service(ServiceManager serviceManager) throws ServiceException 077 { 078 _workspaceSelector = (WorkspaceSelector) serviceManager.lookup(WorkspaceSelector.ROLE); 079 _repository = (JackrabbitRepository) serviceManager.lookup(AbstractRepository.ROLE); 080 } 081 082 @Override 083 public void initialize() throws Exception 084 { 085 _solrUrl = Config.getInstance().getValue(__SOLR_URL_CONFIG); 086 _solrSocketTimeout = Optional.of(__SOLR_SOCKET_TIMEOUT_CONFIG) 087 .map(Config.getInstance()::<Long>getValue) 088 .map(Long::intValue); 089 _solrCorePrefix = Config.getInstance().getValue(__SOLR_CORE_PREFIX_CONFIG); 090 091 Http2SolrClient.Builder solrReadClientBuilder = new Http2SolrClient.Builder(_solrUrl); 092 if (_solrSocketTimeout.isPresent()) 093 { 094 solrReadClientBuilder.withIdleTimeout(_solrSocketTimeout.get(), TimeUnit.SECONDS); 095 } 096 097 // "Disable" the feature entirely 098 System.setProperty("solr.cloud.client.stallTime", Integer.toString(Integer.MAX_VALUE)); 099 100 _solrReadClient = solrReadClientBuilder.build(); 101 102 String[] workspaces = _repository.getWorkspaces(); 103 _solrDefaultUpdateClients = new HashMap<>(); 104 _solrNoAutoCommitUpdateClients = new HashMap<>(); 105 for (String workspaceName : workspaces) 106 { 107 _solrDefaultUpdateClients.put(workspaceName, _createDefaultUpdateClient(workspaceName)); 108 _solrNoAutoCommitUpdateClients.put(workspaceName, _createNoAutoCommitUpdateClient(workspaceName)); 109 } 110 } 111 112 private SolrClient _createDefaultUpdateClient(String workspaceName) 113 { 114 return new DefaultUpdateClient(_solrUrl, _solrSocketTimeout, getCollectionName(workspaceName), 10, 4, getLogger()); 115 } 116 117 private SolrClient _createNoAutoCommitUpdateClient(String workspaceName) 118 { 119 return new NoAutoCommitUpdateClient(_solrUrl, _solrSocketTimeout, getCollectionName(workspaceName), 10, 4, getLogger()); 120 } 121 122 @Override 123 public void dispose() 124 { 125 // Release the solr clients (as a Closeable). 126 IOUtils.closeQuietly(_solrReadClient); 127 _solrReadClient = null; 128 129 for (SolrClient solrDefaultUpdateClient : _solrDefaultUpdateClients.values()) 130 { 131 IOUtils.closeQuietly(solrDefaultUpdateClient); 132 } 133 _solrDefaultUpdateClients.clear(); 134 _solrDefaultUpdateClients = null; 135 136 for (SolrClient solrNoAutoCommitUpdateClient : _solrNoAutoCommitUpdateClients.values()) 137 { 138 IOUtils.closeQuietly(solrNoAutoCommitUpdateClient); 139 } 140 _solrNoAutoCommitUpdateClients.clear(); 141 _solrNoAutoCommitUpdateClients = null; 142 } 143 144 @Override 145 public SolrClient getReadClient() 146 { 147 return _solrReadClient; 148 } 149 150 @Override 151 public SolrClient getUpdateClient(String workspaceName, boolean autoCommit) 152 { 153 Map<String, SolrClient> updateClients = autoCommit ? _solrDefaultUpdateClients : _solrNoAutoCommitUpdateClients; 154 155 SolrClient updateClient = updateClients.get(_nonNullWorkspaceName(workspaceName)); 156 if (updateClient == null) 157 { 158 // Perhaps the workspace was created after initializing this component, try to check if JCR workspace exist 159 try 160 { 161 if (ArrayUtils.contains(_repository.getWorkspaces(), workspaceName)) 162 { 163 updateClient = autoCommit ? _createDefaultUpdateClient(workspaceName) : _createNoAutoCommitUpdateClient(workspaceName); 164 updateClients.put(workspaceName, updateClient); 165 } 166 } 167 catch (RepositoryException e) 168 { 169 getLogger().error("An error occurs while trying to return all JCR workspaces", e); 170 } 171 } 172 173 return updateClient; 174 } 175 176 @Override 177 public String getCollectionName() 178 { 179 return getCollectionName(_workspaceSelector.getWorkspace()); 180 } 181 182 @Override 183 public String getCollectionName(String workspaceName) 184 { 185 return _solrCorePrefix + _nonNullWorkspaceName(workspaceName); 186 } 187 188 private String _nonNullWorkspaceName(String workspaceName) 189 { 190 if (workspaceName == null) 191 { 192 getLogger().debug("Passing null workspace name. Switching to current workspace."); 193 return _workspaceSelector.getWorkspace(); 194 } 195 else 196 { 197 return workspaceName; 198 } 199 } 200}