001    /**
002     * Copyright 2005-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community 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.opensource.org/licenses/ecl2.php
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     */
016    package org.kuali.rice.kew.actions;
017    
018    import java.util.ArrayList;
019    import java.util.List;
020    
021    import org.apache.commons.collections.CollectionUtils;
022    import org.apache.log4j.MDC;
023    import org.kuali.rice.kew.actionrequest.ActionRequestValue;
024    import org.kuali.rice.kew.actionrequest.Recipient;
025    import org.kuali.rice.kew.actiontaken.ActionTakenValue;
026    import org.kuali.rice.kew.api.action.AdHocRevoke;
027    import org.kuali.rice.kew.api.exception.InvalidActionTakenException;
028    import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
029    import org.kuali.rice.kew.api.KewApiConstants;
030    import org.kuali.rice.kim.api.identity.principal.PrincipalContract;
031    
032    
033    /**
034     * The RevokeAdHocApprove revokes the specified AdHoc requests.
035     *
036     * @author Kuali Rice Team (rice.collab@kuali.org)
037     */
038    public class RevokeAdHocAction extends ActionTakenEvent {
039    
040        private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(RevokeAdHocAction.class);
041    
042        private String actionRequestId;
043        private AdHocRevoke revoke;
044    
045        public RevokeAdHocAction(DocumentRouteHeaderValue routeHeader, PrincipalContract principal) {
046            super(KewApiConstants.ACTION_TAKEN_ADHOC_REVOKED_CD, routeHeader, principal);
047        }
048    
049        public RevokeAdHocAction(DocumentRouteHeaderValue routeHeader, PrincipalContract principal, String actionRequestId, String annotation) {
050            super(KewApiConstants.ACTION_TAKEN_ADHOC_REVOKED_CD, routeHeader, principal, annotation);
051            this.actionRequestId = actionRequestId;
052        }
053        
054        public RevokeAdHocAction(DocumentRouteHeaderValue routeHeader, PrincipalContract principal, AdHocRevoke revoke, String annotation) {
055            super(KewApiConstants.ACTION_TAKEN_ADHOC_REVOKED_CD, routeHeader, principal, annotation);
056            this.revoke = revoke;
057        }
058    
059        /* (non-Javadoc)
060         * @see org.kuali.rice.kew.actions.ActionTakenEvent#isActionCompatibleRequest(java.util.List)
061         */
062        @Override
063        public String validateActionRules() {
064            if (!getRouteHeader().isValidActionToTake(getActionPerformedCode())) {
065                return "Revoke adhoc request is not valid on this document";
066            }
067            return "";
068        }
069        
070        @Override
071        public String validateActionRules(List<ActionRequestValue> actionRequests) {
072            return validateActionRules();
073        }
074    
075        /**
076         * Records the approve action.
077         * - Checks to make sure the document status allows the action.
078         * - Checks that the user has not taken a previous action.
079         * - Deactivates the pending requests for this user
080         * - Records the action
081         *
082         * @throws InvalidActionTakenException
083         */
084        public void recordAction() throws InvalidActionTakenException {
085            MDC.put("docId", getRouteHeader().getDocumentId());
086            updateSearchableAttributesIfPossible();
087    
088            String errorMessage = validateActionRules();
089            if (!org.apache.commons.lang.StringUtils.isEmpty(errorMessage)) {
090                throw new InvalidActionTakenException(errorMessage);
091            }
092    
093            LOG.debug("Revoking adhoc request : " + annotation);
094    
095            List<ActionRequestValue> requestsToRevoke = new ArrayList<ActionRequestValue>();
096            List<ActionRequestValue> actionRequests = getActionRequestService().findPendingRootRequestsByDocId(getDocumentId());
097            for (ActionRequestValue actionRequest : actionRequests)
098            {
099                if (matchesActionRequest(revoke, actionRequest))
100                {
101                    requestsToRevoke.add(actionRequest);
102                }
103            }
104            if (requestsToRevoke.isEmpty() && actionRequestId != null) {
105                    throw new InvalidActionTakenException("Failed to revoke action request with id " + actionRequestId +
106                                    ".  ID does not represent a valid ad hoc request!");
107            }
108    
109            Recipient delegator = findDelegatorForActionRequests(actionRequests);
110            LOG.debug("Record the revoke action");
111            ActionTakenValue actionTaken = saveActionTaken(delegator);
112    
113            LOG.debug("Revoke all matching action requests, number of matching requests: " + requestsToRevoke.size());
114            getActionRequestService().deactivateRequests(actionTaken, requestsToRevoke);
115            notifyActionTaken(actionTaken);
116    
117        }
118        
119        /**
120             * Determines if the given action request is an ad hoc request which matches this set of criteria.
121             */
122            protected boolean matchesActionRequest(AdHocRevoke adHocRevokeCommand, ActionRequestValue actionRequest) {
123                    if (!actionRequest.isAdHocRequest()) {
124                            return false;
125                    }               
126                    if (actionRequestId != null) {
127                            return actionRequestId.equals(actionRequest.getActionRequestId());
128                    } else if (adHocRevokeCommand != null) {
129                            boolean principalOrGroupId = !CollectionUtils.isEmpty(adHocRevokeCommand.getPrincipalIds()) || !CollectionUtils.isEmpty(adHocRevokeCommand.getGroupIds());
130                            if (!CollectionUtils.isEmpty(adHocRevokeCommand.getNodeNames()) && !adHocRevokeCommand.getNodeNames().contains(actionRequest.getNodeInstance().getName())) {
131                                    return false;
132                            }
133                            if (actionRequest.isUserRequest() && !CollectionUtils.isEmpty(adHocRevokeCommand.getPrincipalIds())) {
134                                    return adHocRevokeCommand.getPrincipalIds().contains(actionRequest.getPrincipalId());
135                            }
136                            if (actionRequest.isGroupRequest() && !CollectionUtils.isEmpty(adHocRevokeCommand.getGroupIds())) {
137                                    return adHocRevokeCommand.getGroupIds().contains(actionRequest.getGroupId());
138                            }
139                            return !principalOrGroupId;
140                    }
141                    return true;
142            }
143    
144    }