mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-08-14 17:58:59 +00:00
Turned WebScripts registry into an asynchronously-refreshed component
- The size of the object and the nature in which it is read means that it does not really fit into the clustered cache model. We just notify the cluster if the cache entry it needs to be reloaded, serving stale data for a short while - ALF-19982: BM-0013: Reindex: WebScripts registry is repeatedly reset (DeclarativeRegistry) - ALF-19983 BM-0013: Reindex: Solr reindex works very slowly git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@55415 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
@@ -35,13 +35,18 @@
|
||||
<property name="uriIndex" ref="publicapi.index.prototype" />
|
||||
</bean>
|
||||
|
||||
<bean id="publicapi.container" class="org.alfresco.rest.api.PublicApiRepositoryContainer" parent="webscripts.abstractcontainer" init-method="setup">
|
||||
<property name="name"><value>Public Api</value></property>
|
||||
<bean name="publicapi.webScriptsRegistryCache" class="org.alfresco.repo.web.scripts.RegistryAsynchronouslyRefreshedCache" parent="abstractAsynchronouslyRefreshedCache">
|
||||
<property name="registryFactory">
|
||||
<bean class="org.springframework.beans.factory.config.ObjectFactoryCreatingFactoryBean">
|
||||
<property name="targetBeanName"><idref local="publicapi.registry.prototype"/></property>
|
||||
</bean>
|
||||
</property>
|
||||
<property name="retryingTransactionHelper" ref="retryingTransactionHelper" />
|
||||
</bean>
|
||||
|
||||
<bean id="publicapi.container" class="org.alfresco.rest.api.PublicApiRepositoryContainer" parent="webscripts.abstractcontainer" init-method="setup">
|
||||
<property name="name"><value>Public Api</value></property>
|
||||
<property name="webScriptsRegistryCache" ref="publicapi.webScriptsRegistryCache" />
|
||||
<!-- Use the time-limited transaction helper to keep request times to an acceptable duration -->
|
||||
<property name="transactionService" ref="transactionService" />
|
||||
<!-- The transaction helper used to generate error responses must be unlimited -->
|
||||
@@ -53,7 +58,6 @@
|
||||
<property name="scriptProcessorRegistry" ref="webscripts.repo.registry.scriptprocessor" />
|
||||
<property name="descriptorService" ref="DescriptorService" />
|
||||
<property name="tenantAdminService" ref="tenantAdminService" />
|
||||
<property name="webScriptsRegistryCache" ref="publicapi.webScriptsRegistryCache"/>
|
||||
<property name="encryptTempFiles" value="${webscripts.encryptTempFiles}"/>
|
||||
<property name="tempDirectoryName" value="${webscripts.tempDirectoryName}"/>
|
||||
<property name="memoryThreshold" value="${webscripts.memoryThreshold}"/>
|
||||
|
@@ -206,11 +206,7 @@
|
||||
</entry>
|
||||
</map>
|
||||
</property>
|
||||
<property name="registryFactory">
|
||||
<bean class="org.springframework.beans.factory.config.ObjectFactoryCreatingFactoryBean">
|
||||
<property name="targetBeanName"><idref local="webscripts.registry.prototype"/></property>
|
||||
</bean>
|
||||
</property>
|
||||
<property name="webScriptsRegistryCache" ref="webScriptsRegistryCache"/>
|
||||
<!-- Use the time-limited transaction helper to keep request times to an acceptable duration -->
|
||||
<property name="transactionService" ref="transactionService" />
|
||||
<!-- The transaction helper used to generate error responses must be unlimited -->
|
||||
@@ -222,7 +218,6 @@
|
||||
<property name="scriptProcessorRegistry" ref="webscripts.repo.registry.scriptprocessor" />
|
||||
<property name="descriptorService" ref="DescriptorService" />
|
||||
<property name="tenantAdminService" ref="tenantAdminService" />
|
||||
<property name="webScriptsRegistryCache" ref="webScriptsRegistryCache"/>
|
||||
<property name="encryptTempFiles" value="${webscripts.encryptTempFiles}"/>
|
||||
<property name="tempDirectoryName" value="${webscripts.tempDirectoryName}"/>
|
||||
<property name="memoryThreshold" value="${webscripts.memoryThreshold}"/>
|
||||
@@ -235,6 +230,15 @@
|
||||
<property name="uriIndex" ref="webscripts.index.prototype" />
|
||||
</bean>
|
||||
|
||||
<bean name="webScriptsRegistryCache" class="org.alfresco.repo.web.scripts.RegistryAsynchronouslyRefreshedCache" parent="abstractAsynchronouslyRefreshedCache">
|
||||
<property name="registryFactory">
|
||||
<bean class="org.springframework.beans.factory.config.ObjectFactoryCreatingFactoryBean">
|
||||
<property name="targetBeanName"><idref local="webscripts.registry.prototype"/></property>
|
||||
</bean>
|
||||
</property>
|
||||
<property name="retryingTransactionHelper" ref="retryingTransactionHelper" />
|
||||
</bean>
|
||||
|
||||
<bean id="webscripts.js.paging" class="org.alfresco.repo.web.util.paging.Paging" />
|
||||
|
||||
<!-- -->
|
||||
|
@@ -0,0 +1,95 @@
|
||||
/*
|
||||
* Copyright (C) 2005-2013 Alfresco Software Limited.
|
||||
*
|
||||
* This file is part of Alfresco
|
||||
*
|
||||
* Alfresco is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Lesser General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Alfresco is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Lesser General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Lesser General Public License
|
||||
* along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
package org.alfresco.repo.web.scripts;
|
||||
|
||||
import org.alfresco.repo.cache.AbstractAsynchronouslyRefreshedCache;
|
||||
import org.alfresco.repo.transaction.RetryingTransactionHelper;
|
||||
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
|
||||
import org.alfresco.util.PropertyCheck;
|
||||
import org.apache.commons.logging.Log;
|
||||
import org.apache.commons.logging.LogFactory;
|
||||
import org.springframework.beans.factory.InitializingBean;
|
||||
import org.springframework.beans.factory.ObjectFactory;
|
||||
import org.springframework.extensions.webscripts.Registry;
|
||||
|
||||
/**
|
||||
* Asynchronously refreshed cache for repository webscripts.
|
||||
* <p/>
|
||||
* This does not stop gratuitous calls to <i>refresh</i> but will ensure that, once an instance has been created,
|
||||
* a version of the registry is returned even if it is slighly out of date. This can be changed so that it waits
|
||||
* for reset but is probably not required.
|
||||
*
|
||||
* @author Derek Hulley
|
||||
* @since 4.2.0
|
||||
*/
|
||||
public class RegistryAsynchronouslyRefreshedCache extends AbstractAsynchronouslyRefreshedCache<Registry> implements InitializingBean
|
||||
{
|
||||
private static Log logger = LogFactory.getLog(RegistryAsynchronouslyRefreshedCache.class);
|
||||
|
||||
private ObjectFactory<Registry> registryFactory;
|
||||
private RetryingTransactionHelper retryingTransactionHelper;
|
||||
|
||||
/**
|
||||
* @param registryFactory factory for web script registries
|
||||
*/
|
||||
public void setRegistryFactory(ObjectFactory<Registry> registryFactory)
|
||||
{
|
||||
this.registryFactory = registryFactory;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param retryingTransactionHelper the retryingTransactionHelper to set
|
||||
*/
|
||||
public void setRetryingTransactionHelper(RetryingTransactionHelper retryingTransactionHelper)
|
||||
{
|
||||
this.retryingTransactionHelper = retryingTransactionHelper;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Registry buildCache(final String tenantId)
|
||||
{
|
||||
return retryingTransactionHelper.doInTransaction(new RetryingTransactionCallback<Registry>()
|
||||
{
|
||||
@Override
|
||||
public Registry execute() throws Throwable
|
||||
{
|
||||
return doBuildCache(tenantId);
|
||||
}
|
||||
}, true, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* This method is thread safe as per contract of {@link #buildCache(String)}.
|
||||
*/
|
||||
private Registry doBuildCache(String tenantId)
|
||||
{
|
||||
Registry registry = registryFactory.getObject();
|
||||
registry.reset();
|
||||
logger.info("Fetching web script registry for tenant " + tenantId);
|
||||
return registry;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void afterPropertiesSet() throws Exception
|
||||
{
|
||||
PropertyCheck.mandatory(this, "registryFactory", registryFactory);
|
||||
PropertyCheck.mandatory(this, "retryingTransactionHelper", retryingTransactionHelper);
|
||||
super.afterPropertiesSet();
|
||||
}
|
||||
}
|
@@ -18,17 +18,15 @@
|
||||
*/
|
||||
package org.alfresco.repo.web.scripts;
|
||||
|
||||
import org.alfresco.repo.cache.SimpleCache;
|
||||
import org.alfresco.repo.cache.AsynchronouslyRefreshedCache;
|
||||
import org.alfresco.repo.tenant.TenantAdminService;
|
||||
import org.alfresco.repo.tenant.TenantDeployer;
|
||||
import org.alfresco.repo.transaction.RetryingTransactionHelper.RetryingTransactionCallback;
|
||||
import org.alfresco.service.transaction.TransactionService;
|
||||
import org.apache.commons.logging.Log;
|
||||
import org.apache.commons.logging.LogFactory;
|
||||
import org.springframework.beans.factory.ObjectFactory;
|
||||
import org.springframework.extensions.webscripts.Registry;
|
||||
|
||||
|
||||
/**
|
||||
* Tenant-aware Repository (server-tier) container for Web Scripts
|
||||
*
|
||||
@@ -39,31 +37,21 @@ public class TenantRepositoryContainer extends RepositoryContainer implements Te
|
||||
// Logger
|
||||
protected static final Log logger = LogFactory.getLog(TenantRepositoryContainer.class);
|
||||
|
||||
/** Component Dependencies */
|
||||
/* Component Dependencies */
|
||||
protected TenantAdminService tenantAdminService;
|
||||
protected TransactionService transactionService;
|
||||
protected ObjectFactory registryFactory;
|
||||
protected SimpleCache<String, Registry> webScriptsRegistryCache;
|
||||
protected boolean initialized;
|
||||
private AsynchronouslyRefreshedCache<Registry> registryCache;
|
||||
|
||||
/**
|
||||
* @param webScriptsRegistryCache
|
||||
* @param registryCache asynchronously maintained cache for script registries
|
||||
*/
|
||||
public void setWebScriptsRegistryCache(SimpleCache<String, Registry> webScriptsRegistryCache)
|
||||
public void setWebScriptsRegistryCache(AsynchronouslyRefreshedCache<Registry> registryCache)
|
||||
{
|
||||
this.webScriptsRegistryCache = webScriptsRegistryCache;
|
||||
this.registryCache = registryCache;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param registryFactory
|
||||
*/
|
||||
public void setRegistryFactory(ObjectFactory registryFactory)
|
||||
{
|
||||
this.registryFactory = registryFactory;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param tenantAdminService
|
||||
* @param tenantAdminService service to sort out tenant context
|
||||
*/
|
||||
public void setTenantAdminService(TenantAdminService tenantAdminService)
|
||||
{
|
||||
@@ -71,7 +59,7 @@ public class TenantRepositoryContainer extends RepositoryContainer implements Te
|
||||
}
|
||||
|
||||
/**
|
||||
* @param transactionService the transactionService to set
|
||||
* @param transactionService service to give transactions when reading from the container
|
||||
*/
|
||||
public void setTransactionService(TransactionService transactionService)
|
||||
{
|
||||
@@ -79,69 +67,45 @@ public class TenantRepositoryContainer extends RepositoryContainer implements Te
|
||||
this.transactionService = transactionService;
|
||||
}
|
||||
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.web.scripts.AbstractRuntimeContainer#getRegistry()
|
||||
*/
|
||||
@Override
|
||||
public Registry getRegistry()
|
||||
{
|
||||
String tenantDomain = tenantAdminService.getCurrentUserDomain();
|
||||
Registry registry = webScriptsRegistryCache.get(tenantDomain);
|
||||
if (registry == null)
|
||||
Registry registry = registryCache.get();
|
||||
boolean isUpToDate = registryCache.isUpToDate();
|
||||
if (!isUpToDate && logger.isDebugEnabled())
|
||||
{
|
||||
registry = (Registry)registryFactory.getObject();
|
||||
// We only need to reset the registry if the superclass thinks its already initialized
|
||||
if (initialized)
|
||||
{
|
||||
registry.reset();
|
||||
}
|
||||
webScriptsRegistryCache.put(tenantDomain, registry);
|
||||
logger.debug("Retrieved out of date web script registry for tenant " + tenantAdminService.getCurrentUserDomain());
|
||||
}
|
||||
return registry;
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.repo.tenant.TenantDeployer#onEnableTenant()
|
||||
*/
|
||||
@Override
|
||||
public void onEnableTenant()
|
||||
{
|
||||
init();
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.repo.tenant.TenantDeployer#onDisableTenant()
|
||||
*/
|
||||
@Override
|
||||
public void onDisableTenant()
|
||||
{
|
||||
destroy();
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.repo.tenant.TenantDeployer#init()
|
||||
*/
|
||||
@Override
|
||||
public void init()
|
||||
{
|
||||
tenantAdminService.register(this);
|
||||
registryCache.refresh();
|
||||
|
||||
super.reset();
|
||||
|
||||
initialized = true;
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.repo.tenant.TenantDeployer#destroy()
|
||||
*/
|
||||
@Override
|
||||
public void destroy()
|
||||
{
|
||||
webScriptsRegistryCache.remove(tenantAdminService.getCurrentUserDomain());
|
||||
|
||||
initialized = false;
|
||||
registryCache.refresh();
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.web.scripts.AbstractRuntimeContainer#reset()
|
||||
*/
|
||||
@Override
|
||||
public void reset()
|
||||
{
|
||||
|
@@ -29,10 +29,6 @@ public class PublicApiRepositoryContainer extends TenantRepositoryContainer
|
||||
|
||||
/**
|
||||
* Execute script within required level of transaction
|
||||
*
|
||||
* @param scriptReq
|
||||
* @param scriptRes
|
||||
* @throws IOException
|
||||
*/
|
||||
@Override
|
||||
protected void transactionedExecute(final WebScript script, final WebScriptRequest scriptReq, final WebScriptResponse scriptRes)
|
||||
@@ -49,9 +45,7 @@ public class PublicApiRepositoryContainer extends TenantRepositoryContainer
|
||||
super.transactionedExecute(script, scriptReq, scriptRes);
|
||||
}
|
||||
|
||||
/* (non-Javadoc)
|
||||
* @see org.alfresco.web.scripts.RuntimeContainer#executeScript(org.alfresco.web.scripts.WebScriptRequest, org.alfresco.web.scripts.WebScriptResponse, org.alfresco.web.scripts.Authenticator)
|
||||
*/
|
||||
@Override
|
||||
public void executeScript(final WebScriptRequest scriptReq, final WebScriptResponse scriptRes, final Authenticator auth)
|
||||
throws IOException
|
||||
{
|
||||
|
@@ -86,7 +86,8 @@ public class RepositoryContainerTest extends BaseWebScriptTest
|
||||
/**
|
||||
* Person should be current user irrespective of runas user.
|
||||
*/
|
||||
public void testRunAsAdmin() throws Exception {
|
||||
public void testRunAsAdmin() throws Exception
|
||||
{
|
||||
authenticationComponent.setCurrentUser(USER_ONE);
|
||||
|
||||
// No runas specified within our webscript descriptor
|
||||
|
Reference in New Issue
Block a user