RM-1456, disposition lifecycle executor now does not execute cutoff after undo cutoff

git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/modules/recordsmanagement/HEAD@73541 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
Mark Hibbins
2014-06-11 07:29:21 +00:00
parent abcb6e2433
commit f5063cb8eb
7 changed files with 242 additions and 24 deletions

View File

@@ -881,6 +881,10 @@
</properties> </properties>
</aspect> </aspect>
<aspect name="rma:uncutOff">
<title>Uncut Off</title>
</aspect>
<!-- Indicates that an object is being transferred --> <!-- Indicates that an object is being transferred -->
<aspect name="rma:transferring"> <aspect name="rma:transferring">
<title>Transferring</title> <title>Transferring</title>

View File

@@ -18,6 +18,7 @@
*/ */
package org.alfresco.module.org_alfresco_module_rm.action.impl; package org.alfresco.module.org_alfresco_module_rm.action.impl;
import org.alfresco.error.AlfrescoRuntimeException;
import org.alfresco.module.org_alfresco_module_rm.action.RMDispositionActionExecuterAbstractBase; import org.alfresco.module.org_alfresco_module_rm.action.RMDispositionActionExecuterAbstractBase;
import org.alfresco.service.cmr.action.Action; import org.alfresco.service.cmr.action.Action;
import org.alfresco.service.cmr.repository.NodeRef; import org.alfresco.service.cmr.repository.NodeRef;
@@ -37,18 +38,52 @@ public class CutOffAction extends RMDispositionActionExecuterAbstractBase
*/ */
@Override @Override
protected void executeRecordFolderLevelDisposition(Action action, NodeRef recordFolder) protected void executeRecordFolderLevelDisposition(Action action, NodeRef recordFolder)
{
if(checkUncutOffStatus(action, recordFolder))
{ {
// Mark the folder as cut off // Mark the folder as cut off
dispositionService.cutoffDisposableItem(recordFolder); dispositionService.cutoffDisposableItem(recordFolder);
} }
}
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.action.RMDispositionActionExecuterAbstractBase#executeRecordLevelDisposition(org.alfresco.service.cmr.action.Action, org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.action.RMDispositionActionExecuterAbstractBase#executeRecordLevelDisposition(org.alfresco.service.cmr.action.Action, org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override @Override
protected void executeRecordLevelDisposition(Action action, NodeRef record) protected void executeRecordLevelDisposition(Action action, NodeRef record)
{
if(checkUncutOffStatus(action, record))
{ {
// Mark the record as cut off // Mark the record as cut off
dispositionService.cutoffDisposableItem(record); dispositionService.cutoffDisposableItem(record);
} }
} }
/**
* Check if the record or folder has been uncut off. If it has and this cut off action is an
* automated disposition action then the cut off isn't run. If it has and this is a manual
* cut off action then the uncut off aspect is removed prior to the uncut action.
*
* @param action The cut off action
* @param recordOrFolder The record or folder to be cut off
* @return True if the record or folder can be cut off
*/
private boolean checkUncutOffStatus(Action action, NodeRef recordOrFolder)
{
boolean okToCutOff = true;
if(nodeService.hasAspect(recordOrFolder, ASPECT_UNCUT_OFF))
{
if(action.getParameterValue(PARAM_NO_ERROR_CHECK) != null)
{
// this exception stops the cut off disposition schedule action taking place and because we're
// running from the schedule (PARAM_NO_ERROR_CHECK is set) then the exception will not be reported
throw new AlfrescoRuntimeException("Cannot cut off from schedule when uncut off aspect is present");
}
else
{
nodeService.removeAspect(recordOrFolder, ASPECT_UNCUT_OFF);
}
}
return okToCutOff;
}
}

View File

@@ -56,14 +56,16 @@ public class UnCutoffAction extends RMActionExecuterAbstractBase
throw new AlfrescoRuntimeException(I18NUtil.getMessage(MSG_UNDO_NOT_LAST)); throw new AlfrescoRuntimeException(I18NUtil.getMessage(MSG_UNDO_NOT_LAST));
} }
// Remove the cutoff aspect // Remove the cutoff aspect and add the uncutoff aspect
nodeService.removeAspect(actionedUponNodeRef, ASPECT_CUT_OFF); nodeService.removeAspect(actionedUponNodeRef, ASPECT_CUT_OFF);
nodeService.addAspect(actionedUponNodeRef, ASPECT_UNCUT_OFF, null);
if (recordFolderService.isRecordFolder(actionedUponNodeRef)) if (recordFolderService.isRecordFolder(actionedUponNodeRef))
{ {
List<NodeRef> records = recordService.getRecords(actionedUponNodeRef); List<NodeRef> records = recordService.getRecords(actionedUponNodeRef);
for (NodeRef record : records) for (NodeRef record : records)
{ {
nodeService.removeAspect(record, ASPECT_CUT_OFF); nodeService.removeAspect(record, ASPECT_CUT_OFF);
nodeService.addAspect(record, ASPECT_UNCUT_OFF, null);
} }
} }

View File

@@ -95,6 +95,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
* *
* @param nodeService the node service * @param nodeService the node service
*/ */
@Override
public void setNodeService(NodeService nodeService) public void setNodeService(NodeService nodeService)
{ {
this.nodeService = nodeService; this.nodeService = nodeService;
@@ -105,6 +106,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
* *
* @param dictionaryServic the dictionary service * @param dictionaryServic the dictionary service
*/ */
@Override
public void setDictionaryService(DictionaryService dictionaryService) public void setDictionaryService(DictionaryService dictionaryService)
{ {
this.dictionaryService = dictionaryService; this.dictionaryService = dictionaryService;
@@ -252,6 +254,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getDispositionSchedule(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getDispositionSchedule(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public DispositionSchedule getDispositionSchedule(NodeRef nodeRef) public DispositionSchedule getDispositionSchedule(NodeRef nodeRef)
{ {
DispositionSchedule di = null; DispositionSchedule di = null;
@@ -302,6 +305,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getAssociatedDispositionSchedule(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getAssociatedDispositionSchedule(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public DispositionSchedule getAssociatedDispositionSchedule(NodeRef nodeRef) public DispositionSchedule getAssociatedDispositionSchedule(NodeRef nodeRef)
{ {
DispositionSchedule ds = null; DispositionSchedule ds = null;
@@ -400,6 +404,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getDisposableItems(org.alfresco.module.org_alfresco_module_rm.disposition.DispositionSchedule) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getDisposableItems(org.alfresco.module.org_alfresco_module_rm.disposition.DispositionSchedule)
*/ */
@Override
public List<NodeRef> getDisposableItems(DispositionSchedule dispositionSchedule) public List<NodeRef> getDisposableItems(DispositionSchedule dispositionSchedule)
{ {
ParameterCheck.mandatory("dispositionSchedule", dispositionSchedule); ParameterCheck.mandatory("dispositionSchedule", dispositionSchedule);
@@ -528,6 +533,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* *
*/ */
@Override
public DispositionActionDefinition addDispositionActionDefinition( public DispositionActionDefinition addDispositionActionDefinition(
DispositionSchedule schedule, DispositionSchedule schedule,
Map<QName, Serializable> actionDefinitionParams) Map<QName, Serializable> actionDefinitionParams)
@@ -557,6 +563,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#removeDispositionActionDefinition(org.alfresco.module.org_alfresco_module_rm.disposition.DispositionSchedule, org.alfresco.module.org_alfresco_module_rm.disposition.DispositionActionDefinition) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#removeDispositionActionDefinition(org.alfresco.module.org_alfresco_module_rm.disposition.DispositionSchedule, org.alfresco.module.org_alfresco_module_rm.disposition.DispositionActionDefinition)
*/ */
@Override
public void removeDispositionActionDefinition(DispositionSchedule schedule, DispositionActionDefinition actionDefinition) public void removeDispositionActionDefinition(DispositionSchedule schedule, DispositionActionDefinition actionDefinition)
{ {
// check first whether action definitions can be removed // check first whether action definitions can be removed
@@ -579,6 +586,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
* @param actionDefinitionParams Map of parameters to use to update the action definition * @param actionDefinitionParams Map of parameters to use to update the action definition
* @return The updated DispositionActionDefinition * @return The updated DispositionActionDefinition
*/ */
@Override
public DispositionActionDefinition updateDispositionActionDefinition( public DispositionActionDefinition updateDispositionActionDefinition(
DispositionActionDefinition actionDefinition, DispositionActionDefinition actionDefinition,
Map<QName, Serializable> actionDefinitionParams) Map<QName, Serializable> actionDefinitionParams)
@@ -665,6 +673,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#isNextDispositionActionEligible(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#isNextDispositionActionEligible(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public boolean isNextDispositionActionEligible(NodeRef nodeRef) public boolean isNextDispositionActionEligible(NodeRef nodeRef)
{ {
boolean result = false; boolean result = false;
@@ -748,6 +757,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getNextDispositionAction(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getNextDispositionAction(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public DispositionAction getNextDispositionAction(NodeRef nodeRef) public DispositionAction getNextDispositionAction(NodeRef nodeRef)
{ {
DispositionAction result = null; DispositionAction result = null;
@@ -766,6 +776,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getCompletedDispositionActions(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getCompletedDispositionActions(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public List<DispositionAction> getCompletedDispositionActions(NodeRef nodeRef) public List<DispositionAction> getCompletedDispositionActions(NodeRef nodeRef)
{ {
List<ChildAssociationRef> assocs = nodeService.getChildAssocs(nodeRef, ASSOC_DISPOSITION_ACTION_HISTORY, RegexQNamePattern.MATCH_ALL); List<ChildAssociationRef> assocs = nodeService.getChildAssocs(nodeRef, ASSOC_DISPOSITION_ACTION_HISTORY, RegexQNamePattern.MATCH_ALL);
@@ -781,6 +792,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
/** /**
* @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getLastCompletedDispostionAction(org.alfresco.service.cmr.repository.NodeRef) * @see org.alfresco.module.org_alfresco_module_rm.disposition.DispositionService#getLastCompletedDispostionAction(org.alfresco.service.cmr.repository.NodeRef)
*/ */
@Override
public DispositionAction getLastCompletedDispostionAction(NodeRef nodeRef) public DispositionAction getLastCompletedDispostionAction(NodeRef nodeRef)
{ {
DispositionAction result = null; DispositionAction result = null;
@@ -968,6 +980,12 @@ public class DispositionServiceImpl extends ServiceBaseImpl
// apply cut off // apply cut off
applyCutoff(nodeRef); applyCutoff(nodeRef);
// remove uncut off aspect if applied
if(nodeService.hasAspect(nodeRef, ASPECT_UNCUT_OFF))
{
nodeService.removeAspect(nodeRef, ASPECT_UNCUT_OFF);
}
// close the record folder if it isn't already closed! // close the record folder if it isn't already closed!
if (recordFolderService.isRecordFolder(nodeRef) && if (recordFolderService.isRecordFolder(nodeRef) &&
!recordFolderService.isRecordFolderClosed(nodeRef)) !recordFolderService.isRecordFolderClosed(nodeRef))
@@ -991,6 +1009,7 @@ public class DispositionServiceImpl extends ServiceBaseImpl
{ {
AuthenticationUtil.runAsSystem(new RunAsWork<Void>() AuthenticationUtil.runAsSystem(new RunAsWork<Void>()
{ {
@Override
public Void doWork() public Void doWork()
{ {
// Apply the cut off aspect and set cut off date // Apply the cut off aspect and set cut off date

View File

@@ -130,6 +130,9 @@ public interface RecordsManagementModel extends RecordsManagementCustomModel
QName ASPECT_CUT_OFF = QName.createQName(RM_URI, "cutOff"); QName ASPECT_CUT_OFF = QName.createQName(RM_URI, "cutOff");
QName PROP_CUT_OFF_DATE = QName.createQName(RM_URI, "cutOffDate"); QName PROP_CUT_OFF_DATE = QName.createQName(RM_URI, "cutOffDate");
// Uncut off aspect
QName ASPECT_UNCUT_OFF = QName.createQName(RM_URI, "uncutOff");
// Transferred aspect // Transferred aspect
QName ASPECT_TRANSFERRED = QName.createQName(RM_URI, "transferred"); QName ASPECT_TRANSFERRED = QName.createQName(RM_URI, "transferred");

View File

@@ -0,0 +1,56 @@
/*
* Copyright (C) 2005-2014 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.module.org_alfresco_module_rm.model.rma.aspect;
import org.alfresco.module.org_alfresco_module_rm.model.BaseBehaviourBean;
import org.alfresco.repo.copy.CopyBehaviourCallback;
import org.alfresco.repo.copy.CopyDetails;
import org.alfresco.repo.copy.DoNothingCopyBehaviourCallback;
import org.alfresco.repo.policy.annotation.Behaviour;
import org.alfresco.repo.policy.annotation.BehaviourBean;
import org.alfresco.repo.policy.annotation.BehaviourKind;
import org.alfresco.service.namespace.QName;
/**
* rma:uncutoff behaviour bean
*
* @author Mark Hibbins
* @since 2.2
*/
@BehaviourBean
(
defaultType = "rma:uncutOff"
)
public class UncutoffAspect extends BaseBehaviourBean
{
/**
* Copy callback.
*
* Uncutoff aspect should not be copied.
*/
@Behaviour
(
kind = BehaviourKind.CLASS,
policy = "alf:getCopyCallback"
)
public CopyBehaviourCallback getCopyCallback(QName classRef, CopyDetails copyDetails)
{
return new DoNothingCopyBehaviourCallback();
}
}

View File

@@ -22,6 +22,8 @@ import java.io.Serializable;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
import org.alfresco.error.AlfrescoRuntimeException;
import org.alfresco.module.org_alfresco_module_rm.action.RMDispositionActionExecuterAbstractBase;
import org.alfresco.module.org_alfresco_module_rm.action.impl.CompleteEventAction; import org.alfresco.module.org_alfresco_module_rm.action.impl.CompleteEventAction;
import org.alfresco.module.org_alfresco_module_rm.action.impl.CutOffAction; import org.alfresco.module.org_alfresco_module_rm.action.impl.CutOffAction;
import org.alfresco.module.org_alfresco_module_rm.test.util.BaseRMTestCase; import org.alfresco.module.org_alfresco_module_rm.test.util.BaseRMTestCase;
@@ -47,6 +49,7 @@ public class CutOffTest extends BaseRMTestCase
{ {
NodeRef recordFolder = null; NodeRef recordFolder = null;
@Override
public void given() public void given()
{ {
//create record folder //create record folder
@@ -60,12 +63,14 @@ public class CutOffTest extends BaseRMTestCase
rmActionService.executeRecordsManagementAction(recordFolder, CompleteEventAction.NAME, params); rmActionService.executeRecordsManagementAction(recordFolder, CompleteEventAction.NAME, params);
} }
@Override
public void when() public void when()
{ {
// complete event // complete event
rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, null); rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, null);
} }
@Override
public void then() public void then()
{ {
// ensure the record folder is cut off // ensure the record folder is cut off
@@ -87,6 +92,7 @@ public class CutOffTest extends BaseRMTestCase
{ {
NodeRef recordFolder = null; NodeRef recordFolder = null;
@Override
public void given() public void given()
{ {
//create record folder //create record folder
@@ -103,12 +109,14 @@ public class CutOffTest extends BaseRMTestCase
recordFolderService.closeRecordFolder(recordFolder); recordFolderService.closeRecordFolder(recordFolder);
} }
@Override
public void when() public void when()
{ {
// complete event // complete event
rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, null); rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, null);
} }
@Override
public void then() public void then()
{ {
// ensure the record folder is cut off // ensure the record folder is cut off
@@ -116,4 +124,95 @@ public class CutOffTest extends BaseRMTestCase
} }
}); });
} }
/**
* given we have a record folder that is eligible for cutoff ensure that the
* record can be cut off successfully.
*/
public void testCutOffUncutOffRecordFolder()
{
doBehaviourDrivenTest(new BehaviourDrivenTest()
{
NodeRef recordFolder = null;
@Override
public void given()
{
//create record folder
recordFolder = recordFolderService.createRecordFolder(rmContainer, GUID.generate());
nodeService.addAspect(recordFolder, ASPECT_UNCUT_OFF, null);
// TODO add some records
// make eligible for cutoff
Map<String, Serializable> params = new HashMap<String, Serializable>(1);
params.put(CompleteEventAction.PARAM_EVENT_NAME, CommonRMTestUtils.DEFAULT_EVENT_NAME);
rmActionService.executeRecordsManagementAction(recordFolder, CompleteEventAction.NAME, params);
}
@Override
public void when()
{
// complete event
rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, null);
}
@Override
public void then()
{
// ensure the record folder is cut off
assertTrue(dispositionService.isDisposableItemCutoff(recordFolder));
}
});
}
/**
* given we have a record folder that is eligible for cutoff ensure that the
* record can be cut off successfully.
*/
public void testCutOffUncutOffRecordFolderFromSchedule()
{
doBehaviourDrivenTest(new BehaviourDrivenTest()
{
NodeRef recordFolder = null;
@Override
public void given()
{
//create record folder
recordFolder = recordFolderService.createRecordFolder(rmContainer, GUID.generate());
nodeService.addAspect(recordFolder, ASPECT_UNCUT_OFF, null);
// TODO add some records
// make eligible for cutoff
Map<String, Serializable> params = new HashMap<String, Serializable>(1);
params.put(CompleteEventAction.PARAM_EVENT_NAME, CommonRMTestUtils.DEFAULT_EVENT_NAME);
rmActionService.executeRecordsManagementAction(recordFolder, CompleteEventAction.NAME, params);
}
@Override
public void when()
{
// complete event
Map<String, Serializable> params = new HashMap<String, Serializable>(1);
params.put(RMDispositionActionExecuterAbstractBase.PARAM_NO_ERROR_CHECK, Boolean.FALSE);
try
{
rmActionService.executeRecordsManagementAction(recordFolder, CutOffAction.NAME, params);
}
catch(AlfrescoRuntimeException e) { } // expected
}
@Override
public void then()
{
// ensure the record folder is cut off
assertFalse(dispositionService.isDisposableItemCutoff(recordFolder));
}
});
}
} }