/* * 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 . */ package org.alfresco.repo.action.executer; import java.util.HashSet; import java.util.Set; import org.alfresco.repo.action.ActionDefinitionImpl; import org.alfresco.repo.action.ParameterizedItemAbstractBase; import org.alfresco.repo.lock.LockUtils; import org.alfresco.service.cmr.action.Action; import org.alfresco.service.cmr.action.ActionDefinition; import org.alfresco.service.cmr.dictionary.DictionaryService; import org.alfresco.service.cmr.lock.LockService; import org.alfresco.service.cmr.repository.NodeRef; import org.alfresco.service.cmr.repository.NodeService; import org.alfresco.service.namespace.QName; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; /** * Rule action executor abstract base. * * @author Roy Wetherall */ public abstract class ActionExecuterAbstractBase extends ParameterizedItemAbstractBase implements ActionExecuter { private static Log logger = LogFactory.getLog(ActionExecuterAbstractBase.class); protected ActionDefinition actionDefinition; private LockService lockService; private NodeService baseNodeService; private DictionaryService dictionaryService; private NodeService mlAwareNodeService; /** Indicate if the action status should be tracked or not (default false) */ private boolean trackStatus = false; /** Indicated whether the action is public or internal (default true) */ protected boolean publicAction = true; /** List of types and aspects for which this action is applicable */ protected Set applicableTypes = new HashSet(); /** Default queue name */ private String queueName = ""; /** Indicates whether the action should be ignored if the actioned upon node is locked */ private boolean ignoreLock = true; /** * Init method */ public void init() { if (this.publicAction == true) { this.runtimeActionService.registerActionExecuter(this); } } public void setMlAwareNodeService(NodeService mlAwareNodeService) { this.mlAwareNodeService = mlAwareNodeService; } public void setLockService(LockService lockService) { this.lockService = lockService; } public void setBaseNodeService(NodeService nodeService) { this.baseNodeService = nodeService; } /** * Set the dictionary service * * @param dictionaryService the dictionary service */ public void setDictionaryService(DictionaryService dictionaryService) { this.dictionaryService = dictionaryService; } /** * Set whether the action is public or not. * * @param publicAction true if the action is public, false otherwise */ public void setPublicAction(boolean publicAction) { this.publicAction = publicAction; } /** * {@inheritDoc} */ @Override public boolean getTrackStatus() { return trackStatus; } /** * Set whether the basic action definition requires status tracking. * This can be overridden on each action instance but if not, it falls back * to this definition. *

* Setting this to true introduces performance problems for concurrently-executing * rules on V3.4: ALF-7341. * It should only be used for long, seldom-run actions. * * @param trackStatus true to track execution status otherwise false * * @since 3.4.1 */ public void setTrackStatus(boolean trackStatus) { this.trackStatus = trackStatus; } /** * Set the list of types for which this action is applicable * * @param applicableTypes array of applicable types */ public void setApplicableTypes(String[] applicableTypes) { for (String type : applicableTypes) { this.applicableTypes.add(QName.createQName(type)); } } /** * @see org.alfresco.repo.action.executer.ActionExecuter#getIgnoreLock() */ public boolean getIgnoreLock() { return this.ignoreLock; } /** * Set the ignore lock value. * @param ignoreLock true if lock should be ignored on actioned upon node, false otherwise */ public void setIgnoreLock(boolean ignoreLock) { this.ignoreLock = ignoreLock; } /** * Check if a node is a type or subtype of the of one of the applicable types * * @param actionedUponNodeRef the node to check * @return Returns true if the node is in the list of * {@link #setApplicableTypes(String[]) applicable types} or one of the * subtypes */ protected boolean isApplicableType(NodeRef actionedUponNodeRef) { if (this.baseNodeService.exists(actionedUponNodeRef) == true) { QName nodeType = baseNodeService.getType(actionedUponNodeRef); // Quick check in the set if (applicableTypes.contains(nodeType)) { return true; } else { // Have to do a long-winded check for (QName type : applicableTypes) { if (this.dictionaryService.isSubClass(nodeType, type)) { return true; } // Not a subtype; keep checking } } } return false; } /** * Get rule action definition * * @return the action definition object */ public ActionDefinition getActionDefinition() { if (this.actionDefinition == null) { this.actionDefinition = createActionDefinition(this.name); ((ActionDefinitionImpl)this.actionDefinition).setTitleKey(getTitleKey()); ((ActionDefinitionImpl)this.actionDefinition).setDescriptionKey(getDescriptionKey()); ((ActionDefinitionImpl)this.actionDefinition).setTrackStatus(getTrackStatus()); ((ActionDefinitionImpl)this.actionDefinition).setAdhocPropertiesAllowed(getAdhocPropertiesAllowed()); ((ActionDefinitionImpl)this.actionDefinition).setRuleActionExecutor(this.name); ((ActionDefinitionImpl)this.actionDefinition).setParameterDefinitions(getParameterDefintions()); ((ActionDefinitionImpl)this.actionDefinition).setApplicableTypes(this.applicableTypes); } return this.actionDefinition; } /** * This method returns an instance of an ActionDefinition implementation class. By default * this will be an {@link ActionDefinitionImpl}, but this could be overridden. */ protected ActionDefinition createActionDefinition(String name) { return new ActionDefinitionImpl(name); } /** * {@inheritDoc} */ public void execute(Action action, NodeRef actionedUponNodeRef) { // Check the mandatory properties checkMandatoryProperties(action, getActionDefinition()); // Only execute the action if this action is read only or the actioned upon node reference doesn't // have a lock applied for this user. boolean nodeIsLockedForThisUser = false; // null nodeRefs can't be locked and some actions can be run against 'null' nodes. // non-existent nodes can't be locked. if (!ignoreLock && actionedUponNodeRef != null && mlAwareNodeService.exists(actionedUponNodeRef)) { nodeIsLockedForThisUser = LockUtils.isLockedAndReadOnly(actionedUponNodeRef, lockService); } if ( !nodeIsLockedForThisUser) { // Execute the implementation executeImpl(action, actionedUponNodeRef); } else { if (logger.isWarnEnabled() == true) { logger.warn("Action (" + action.getActionDefinitionName() + ") ignored because actioned upon node (" + actionedUponNodeRef + ") is locked."); } } } /** * Execute the action implementation * * @param action the action * @param actionedUponNodeRef the actioned upon node */ protected abstract void executeImpl(Action action, NodeRef actionedUponNodeRef); /** * Set the queueName which will execute this action * if blank or null then the action will be executed on the "default" queue * @param the name of the execution queue which should execute this action. */ public void setQueueName(String queueName) { this.queueName = queueName; } public String getQueueName() { return queueName; } }