| 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 |  |  } |