1 | |
|
2 | |
|
3 | |
|
4 | |
|
5 | |
|
6 | |
|
7 | |
|
8 | |
|
9 | |
|
10 | |
|
11 | |
|
12 | |
|
13 | |
|
14 | |
|
15 | |
|
16 | |
package org.kuali.student.lum.workflow; |
17 | |
import java.util.List; |
18 | |
|
19 | |
import javax.xml.namespace.QName; |
20 | |
|
21 | |
import org.apache.commons.lang.StringUtils; |
22 | |
import org.kuali.rice.core.resourceloader.GlobalResourceLoader; |
23 | |
import org.kuali.rice.kew.actionrequest.ActionRequestValue; |
24 | |
import org.kuali.rice.kew.actiontaken.ActionTakenValue; |
25 | |
import org.kuali.rice.kew.dto.ActionRequestDTO; |
26 | |
import org.kuali.rice.kew.postprocessor.ActionTakenEvent; |
27 | |
import org.kuali.rice.kew.postprocessor.AfterProcessEvent; |
28 | |
import org.kuali.rice.kew.postprocessor.BeforeProcessEvent; |
29 | |
import org.kuali.rice.kew.postprocessor.DeleteEvent; |
30 | |
import org.kuali.rice.kew.postprocessor.DocumentLockingEvent; |
31 | |
import org.kuali.rice.kew.postprocessor.DocumentRouteLevelChange; |
32 | |
import org.kuali.rice.kew.postprocessor.DocumentRouteStatusChange; |
33 | |
import org.kuali.rice.kew.postprocessor.IDocumentEvent; |
34 | |
import org.kuali.rice.kew.postprocessor.PostProcessor; |
35 | |
import org.kuali.rice.kew.postprocessor.ProcessDocReport; |
36 | |
import org.kuali.rice.kew.service.KEWServiceLocator; |
37 | |
import org.kuali.rice.kew.service.WorkflowDocument; |
38 | |
import org.kuali.rice.kew.util.KEWConstants; |
39 | |
import org.kuali.rice.kim.bo.entity.dto.KimPrincipalInfo; |
40 | |
import org.kuali.rice.kim.bo.types.dto.AttributeSet; |
41 | |
import org.kuali.rice.kim.service.KIMServiceLocator; |
42 | |
import org.kuali.rice.student.StudentWorkflowConstants; |
43 | |
import org.kuali.rice.student.bo.KualiStudentKimAttributes; |
44 | |
import org.kuali.student.core.exceptions.OperationFailedException; |
45 | |
import org.kuali.student.core.proposal.ProposalConstants; |
46 | |
import org.kuali.student.core.proposal.dto.ProposalInfo; |
47 | |
import org.kuali.student.core.proposal.service.ProposalService; |
48 | |
import org.kuali.student.core.rice.StudentIdentityConstants; |
49 | 0 | public class KualiStudentPostProcessorBase implements PostProcessor{ |
50 | 0 | private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(KualiStudentPostProcessorBase.class); |
51 | |
|
52 | |
private ProposalService proposalService; |
53 | |
|
54 | |
public ProcessDocReport afterProcess(AfterProcessEvent arg0) throws Exception { |
55 | 0 | return new ProcessDocReport(true); |
56 | |
} |
57 | |
|
58 | |
public ProcessDocReport beforeProcess(BeforeProcessEvent arg0) throws Exception { |
59 | 0 | return new ProcessDocReport(true); |
60 | |
} |
61 | |
|
62 | |
public ProcessDocReport doActionTaken(ActionTakenEvent actionTakenEvent) throws Exception { |
63 | 0 | ActionTakenValue actionTaken = KEWServiceLocator.getActionTakenService().findByActionTakenId(actionTakenEvent.getActionTaken().getActionTakenId()); |
64 | 0 | if (actionTaken == null) { |
65 | 0 | if (LOG.isInfoEnabled()) { |
66 | 0 | LOG.info("Could not find valid ActionTakenValue for doc id '" + actionTakenEvent.getRouteHeaderId() + "'" + |
67 | |
((actionTakenEvent.getActionTaken() == null) ? "" : " for action: " + actionTakenEvent.getActionTaken().getActionTakenLabel())); |
68 | |
} |
69 | 0 | actionTaken = actionTakenEvent.getActionTaken(); |
70 | |
} |
71 | 0 | boolean success = true; |
72 | |
|
73 | 0 | if (!StringUtils.equals(KEWConstants.ROUTE_HEADER_SAVED_CD, actionTaken.getActionTaken())) { |
74 | 0 | ProposalInfo proposalInfo = getProposalService().getProposalByWorkflowId(actionTakenEvent.getRouteHeaderId().toString()); |
75 | 0 | if (actionTaken == null) { |
76 | 0 | throw new OperationFailedException("No action taken found for document id " + actionTakenEvent.getRouteHeaderId()); |
77 | |
} |
78 | 0 | if (StringUtils.equals(KEWConstants.ACTION_TAKEN_SU_DISAPPROVED_CD, actionTaken.getActionTaken())) { |
79 | |
|
80 | 0 | processSuperUserDisapproveActionTaken(actionTakenEvent, actionTaken, proposalInfo); |
81 | |
} |
82 | |
|
83 | 0 | else if (!StringUtils.equals(KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD, actionTaken.getActionTaken())) { |
84 | 0 | for (ActionRequestValue actionRequest : actionTaken.getActionRequests()) { |
85 | 0 | if (actionRequest.isAdHocRequest() && actionRequest.isUserRequest()) { |
86 | 0 | processActionTakenOnAdhocRequest(actionTakenEvent, actionRequest); |
87 | |
} |
88 | |
} |
89 | |
} |
90 | 0 | success = processCustomActionTaken(actionTakenEvent, actionTaken, proposalInfo); |
91 | 0 | } else { |
92 | 0 | success = processCustomSaveActionTaken(actionTakenEvent, actionTaken); |
93 | |
} |
94 | 0 | return new ProcessDocReport(success); |
95 | |
} |
96 | |
|
97 | |
protected boolean processCustomActionTaken(ActionTakenEvent actionTakenEvent, ActionTakenValue actionTaken, ProposalInfo proposalInfo) throws Exception { |
98 | |
|
99 | 0 | return true; |
100 | |
} |
101 | |
|
102 | |
protected boolean processCustomSaveActionTaken(ActionTakenEvent actionTakenEvent, ActionTakenValue actionTaken) throws Exception { |
103 | |
|
104 | 0 | return true; |
105 | |
} |
106 | |
|
107 | |
protected void processActionTakenOnAdhocRequest(ActionTakenEvent actionTakenEvent, ActionRequestValue actionRequestValue) throws Exception { |
108 | 0 | WorkflowDocument doc = new WorkflowDocument(getPrincipalIdForSystemUser(), actionTakenEvent.getRouteHeaderId()); |
109 | 0 | LOG.info("Clearing EDIT permissions added via adhoc requests to principal id: " + actionRequestValue.getPrincipalId()); |
110 | 0 | removeEditAdhocPermissions(actionRequestValue.getPrincipalId(), doc); |
111 | 0 | } |
112 | |
|
113 | |
protected void processSuperUserDisapproveActionTaken(ActionTakenEvent actionTakenEvent, ActionTakenValue actionTaken, ProposalInfo proposalInfo) throws Exception { |
114 | 0 | LOG.info("Action taken was 'Super User Disapprove' which is a 'Withdraw' in Kuali Student"); |
115 | 0 | LOG.info("Will set proposal state to '" + ProposalConstants.PROPOSAL_STATE_WITHDRAWN + "'"); |
116 | 0 | updateProposal(actionTakenEvent, ProposalConstants.PROPOSAL_STATE_WITHDRAWN, proposalInfo); |
117 | 0 | processWithdrawActionTaken(actionTakenEvent, proposalInfo); |
118 | 0 | } |
119 | |
|
120 | |
protected void processWithdrawActionTaken(ActionTakenEvent actionTakenEvent, ProposalInfo proposalInfo) throws Exception { |
121 | |
|
122 | 0 | } |
123 | |
|
124 | |
public ProcessDocReport doDeleteRouteHeader(DeleteEvent arg0) throws Exception { |
125 | 0 | return new ProcessDocReport(true); |
126 | |
} |
127 | |
|
128 | |
public ProcessDocReport doRouteLevelChange(DocumentRouteLevelChange documentRouteLevelChange) throws Exception { |
129 | 0 | ProposalInfo proposalInfo = getProposalService().getProposalByWorkflowId(documentRouteLevelChange.getRouteHeaderId().toString()); |
130 | |
|
131 | |
|
132 | 0 | if (StringUtils.equals(StudentWorkflowConstants.DEFAULT_WORKFLOW_DOCUMENT_START_NODE_NAME,documentRouteLevelChange.getOldNodeName())) { |
133 | |
|
134 | 0 | WorkflowDocument doc = new WorkflowDocument(getPrincipalIdForSystemUser(), documentRouteLevelChange.getRouteHeaderId()); |
135 | 0 | for (ActionRequestDTO actionRequestDTO : doc.getActionRequests()) { |
136 | 0 | if (actionRequestDTO.isAdHocRequest() && actionRequestDTO.isUserRequest() && |
137 | |
StringUtils.equals(documentRouteLevelChange.getOldNodeName(),actionRequestDTO.getNodeName())) { |
138 | 0 | LOG.info("Clearing EDIT permissions added via adhoc requests to principal id: " + actionRequestDTO.getPrincipalId()); |
139 | 0 | removeEditAdhocPermissions(actionRequestDTO.getPrincipalId(), doc); |
140 | |
} |
141 | |
} |
142 | 0 | } |
143 | |
else { |
144 | 0 | LOG.warn("Will not clear any permissions added via adhoc requests"); |
145 | |
} |
146 | 0 | boolean success = processCustomRouteLevelChange(documentRouteLevelChange, proposalInfo); |
147 | 0 | return new ProcessDocReport(success); |
148 | |
} |
149 | |
|
150 | |
protected boolean processCustomRouteLevelChange(DocumentRouteLevelChange documentRouteLevelChange, ProposalInfo proposalInfo) throws Exception { |
151 | |
|
152 | 0 | return true; |
153 | |
} |
154 | |
|
155 | |
public ProcessDocReport doRouteStatusChange(DocumentRouteStatusChange statusChangeEvent) throws Exception { |
156 | 0 | boolean success = true; |
157 | |
|
158 | 0 | if (StringUtils.equals(KEWConstants.ROUTE_HEADER_INITIATED_CD, statusChangeEvent.getOldRouteStatus()) && |
159 | |
StringUtils.equals(KEWConstants.ROUTE_HEADER_SAVED_CD, statusChangeEvent.getNewRouteStatus())) { |
160 | |
|
161 | 0 | success = processCustomRouteStatusSavedStatusChange(statusChangeEvent); |
162 | |
} else { |
163 | 0 | ProposalInfo proposalInfo = getProposalService().getProposalByWorkflowId(statusChangeEvent.getRouteHeaderId().toString()); |
164 | |
|
165 | |
|
166 | 0 | String proposalState = getProposalStateForRouteStatus(proposalInfo.getState(), statusChangeEvent.getNewRouteStatus()); |
167 | 0 | updateProposal(statusChangeEvent, proposalState, proposalInfo); |
168 | 0 | success = processCustomRouteStatusChange(statusChangeEvent, proposalInfo); |
169 | |
} |
170 | 0 | return new ProcessDocReport(success); |
171 | |
} |
172 | |
|
173 | |
protected boolean processCustomRouteStatusChange(DocumentRouteStatusChange statusChangeEvent, ProposalInfo proposalInfo) throws Exception { |
174 | |
|
175 | 0 | return true; |
176 | |
} |
177 | |
|
178 | |
protected boolean processCustomRouteStatusSavedStatusChange(DocumentRouteStatusChange statusChangeEvent) throws Exception { |
179 | |
|
180 | 0 | return true; |
181 | |
} |
182 | |
|
183 | |
public List<Long> getDocumentIdsToLock(DocumentLockingEvent arg0) throws Exception { |
184 | 0 | return null; |
185 | |
} |
186 | |
|
187 | |
|
188 | |
|
189 | |
|
190 | |
|
191 | |
|
192 | |
protected String getProposalStateForRouteStatus(String currentProposalState, String newWorkflowStatusCode) { |
193 | 0 | if (StringUtils.equals(ProposalConstants.PROPOSAL_STATE_WITHDRAWN, currentProposalState)) { |
194 | |
|
195 | 0 | return null; |
196 | |
} |
197 | 0 | if (StringUtils.equals(KEWConstants.ROUTE_HEADER_SAVED_CD, newWorkflowStatusCode)) { |
198 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_SAVED); |
199 | 0 | } else if (KEWConstants.ROUTE_HEADER_ENROUTE_CD.equals(newWorkflowStatusCode)) { |
200 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_ENROUTE); |
201 | 0 | } else if (KEWConstants.ROUTE_HEADER_CANCEL_CD .equals(newWorkflowStatusCode)) { |
202 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_CANCELLED); |
203 | 0 | } else if (KEWConstants.ROUTE_HEADER_DISAPPROVED_CD.equals(newWorkflowStatusCode)) { |
204 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_REJECTED); |
205 | 0 | } else if (KEWConstants.ROUTE_HEADER_PROCESSED_CD.equals(newWorkflowStatusCode)) { |
206 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_APPROVED); |
207 | 0 | } else if (KEWConstants.ROUTE_HEADER_EXCEPTION_CD.equals(newWorkflowStatusCode)) { |
208 | 0 | return getProposalStateFromNewState(currentProposalState, ProposalConstants.PROPOSAL_STATE_EXCEPTION); |
209 | |
} else { |
210 | |
|
211 | 0 | return null; |
212 | |
} |
213 | |
} |
214 | |
|
215 | |
|
216 | |
|
217 | |
|
218 | |
|
219 | |
protected String getProposalStateFromNewState(String currentProposalState, String newProposalState) { |
220 | 0 | if (LOG.isInfoEnabled()) { |
221 | 0 | LOG.info("current proposal state is '" + currentProposalState + "' and new proposal state will be '" + newProposalState + "'"); |
222 | |
} |
223 | 0 | return getStateFromNewState(currentProposalState, newProposalState); |
224 | |
} |
225 | |
|
226 | |
|
227 | |
|
228 | |
|
229 | |
|
230 | |
protected String getStateFromNewState(String currentState, String newState) { |
231 | 0 | if (StringUtils.equals(currentState, newState)) { |
232 | 0 | if (LOG.isInfoEnabled()) { |
233 | 0 | LOG.info("returning null as current state and new state are both '" + currentState + "'"); |
234 | |
} |
235 | 0 | return null; |
236 | |
} |
237 | 0 | return newState; |
238 | |
} |
239 | |
|
240 | |
protected void removeEditAdhocPermissions(String principalId, WorkflowDocument doc) { |
241 | 0 | AttributeSet qualifications = new AttributeSet(); |
242 | 0 | qualifications.put(KualiStudentKimAttributes.DOCUMENT_TYPE_NAME,doc.getDocumentType()); |
243 | 0 | qualifications.put(KualiStudentKimAttributes.QUALIFICATION_DATA_ID,doc.getAppDocId()); |
244 | 0 | KIMServiceLocator.getRoleManagementService().removePrincipalFromRole(principalId, StudentWorkflowConstants.ROLE_NAME_ADHOC_EDIT_PERMISSIONS_ROLE_NAMESPACE, StudentWorkflowConstants.ROLE_NAME_ADHOC_EDIT_PERMISSIONS_ROLE_NAME, qualifications); |
245 | 0 | } |
246 | |
|
247 | |
protected void removeCommentAdhocPermissions(String roleNamespace, String roleName, String principalId, WorkflowDocument doc) { |
248 | 0 | AttributeSet qualifications = new AttributeSet(); |
249 | 0 | qualifications.put(KualiStudentKimAttributes.DOCUMENT_TYPE_NAME,doc.getDocumentType()); |
250 | 0 | qualifications.put(KualiStudentKimAttributes.QUALIFICATION_DATA_ID,doc.getAppDocId()); |
251 | 0 | KIMServiceLocator.getRoleManagementService().removePrincipalFromRole(principalId, StudentWorkflowConstants.ROLE_NAME_ADHOC_ADD_COMMENT_PERMISSIONS_ROLE_NAMESPACE, StudentWorkflowConstants.ROLE_NAME_ADHOC_ADD_COMMENT_PERMISSIONS_ROLE_NAME, qualifications); |
252 | 0 | } |
253 | |
|
254 | |
protected String getPrincipalIdForSystemUser() { |
255 | 0 | KimPrincipalInfo principal = KIMServiceLocator.getIdentityManagementService().getPrincipalByPrincipalName(StudentIdentityConstants.SYSTEM_USER_PRINCIPAL_NAME); |
256 | 0 | if (principal == null) { |
257 | 0 | throw new RuntimeException("Cannot find Principal for principal name: " + StudentIdentityConstants.SYSTEM_USER_PRINCIPAL_NAME); |
258 | |
} |
259 | 0 | return principal.getPrincipalId(); |
260 | |
} |
261 | |
|
262 | |
protected void updateProposal(IDocumentEvent iDocumentEvent, String proposalState, ProposalInfo proposalInfo) throws Exception { |
263 | 0 | if (LOG.isInfoEnabled()) { |
264 | 0 | LOG.info("Setting state '" + proposalState + "' on Proposal with docId='" + proposalInfo.getWorkflowId() + "' and proposalId='" + proposalInfo.getId() + "'"); |
265 | |
} |
266 | 0 | boolean requiresSave = false; |
267 | 0 | if (proposalState != null) { |
268 | 0 | proposalInfo.setState(proposalState); |
269 | 0 | requiresSave = true; |
270 | |
} |
271 | 0 | requiresSave |= preProcessProposalSave(iDocumentEvent, proposalInfo); |
272 | 0 | if (requiresSave) { |
273 | 0 | getProposalService().updateProposal(proposalInfo.getId(), proposalInfo); |
274 | |
} |
275 | 0 | } |
276 | |
|
277 | |
protected boolean preProcessProposalSave(IDocumentEvent iDocumentEvent, ProposalInfo proposalInfo) { |
278 | 0 | return false; |
279 | |
} |
280 | |
|
281 | |
protected ProposalService getProposalService() { |
282 | 0 | if (this.proposalService == null) { |
283 | 0 | this.proposalService = (ProposalService) GlobalResourceLoader.getService(new QName("http://student.kuali.org/wsdl/proposal","ProposalService")); |
284 | |
} |
285 | 0 | return this.proposalService; |
286 | |
} |
287 | |
|
288 | |
} |