001    /**
002     * Copyright 2005-2012 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.ken.deliverer.impl;
017    
018    import java.io.ByteArrayOutputStream;
019    import java.io.IOException;
020    import java.util.Properties;
021    
022    import org.kuali.rice.ken.bo.NotificationMessageDelivery;
023    import org.kuali.rice.ken.core.GlobalNotificationServiceLocator;
024    import org.kuali.rice.ken.deliverer.NotificationMessageDeliverer;
025    import org.kuali.rice.ken.exception.NotificationAutoRemoveException;
026    import org.kuali.rice.ken.exception.NotificationMessageDeliveryException;
027    import org.kuali.rice.ken.service.NotificationWorkflowDocumentService;
028    import org.kuali.rice.ken.util.NotificationConstants;
029    import org.kuali.rice.ken.util.Util;
030    import org.kuali.rice.kew.api.WorkflowDocument;
031    
032    /**
033     * This class is responsible for describing the default delivery mechanism for the system - the KEW
034     * Action List.
035     * @author Kuali Rice Team (rice.collab@kuali.org)
036     */
037    public class KEWActionListMessageDeliverer implements NotificationMessageDeliverer {
038        private static org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(KEWActionListMessageDeliverer.class);
039    
040        /**
041         * Property set in the attribute content that indicates the action received by workflow was
042         * initiated by the Notification System itself (and not an end user)
043         */
044        public static final String INTERNAL_COMMAND_FLAG = "internal_command";
045    
046        private NotificationWorkflowDocumentService notificationWorkflowDocumentService;
047    
048        /**
049         * Constructs a KEWActionListMessageDeliverer.java.
050         */
051        public KEWActionListMessageDeliverer() {
052            this.notificationWorkflowDocumentService = GlobalNotificationServiceLocator.getInstance()
053                    .getNotificationWorkflowDocumentService();
054        }
055    
056        /**
057         * This implementation leverages the workflow integration services to push this notification
058         * into the KEW action list.
059         * @see org.kuali.rice.ken.deliverer.NotificationMessageDeliverer#deliverMessage(org.kuali.rice.ken.bo.NotificationMessageDelivery)
060         */
061        public void deliverMessage(NotificationMessageDelivery messageDelivery) throws NotificationMessageDeliveryException {
062            // make the call to actually generate and ad-hoc route a workflow document
063            String documentId = notificationWorkflowDocumentService.createAndAdHocRouteNotificationWorkflowDocument(
064                    messageDelivery,
065                    Util.getNotificationSystemUser(),
066                    messageDelivery.getUserRecipientId(),
067                    NotificationConstants.KEW_CONSTANTS.GENERIC_DELIVERY_ANNOTATION);
068    
069            // now set the workflow doc id into the message delivery's delivery system id
070            messageDelivery.setDeliverySystemId(documentId);
071            LOG.debug("Message Delivery: " + messageDelivery.toString());
072        }
073    
074        /**
075         * This implementation does an auto-remove by "canceling" the workflow document associated with
076         * the message delivery record. This prevents the user from seeing the item in their list
077         * anymore.
078         * @see org.kuali.rice.ken.deliverer.NotificationMessageDeliverer#autoRemoveMessageDelivery(org.kuali.rice.ken.bo.NotificationMessageDelivery)
079         */
080        public void autoRemoveMessageDelivery(NotificationMessageDelivery messageDelivery)
081                throws NotificationAutoRemoveException {
082            // first retrieve the appropriate notification workflow document to "auto-remove" and proxy as the recipient
083            WorkflowDocument workflowDoc = null;
084            String sysId = messageDelivery.getDeliverySystemId();
085            if (sysId == null) {
086                LOG.error("NotificationMessageDelivery " + messageDelivery.getId()
087                        + " is missing delivery system id (workflow document id");
088                // there is no possibility for recovery, so since there is no id, we'll just log an error and return successfully instead
089                // of throwing an exception
090                return;
091            }
092    
093            workflowDoc = notificationWorkflowDocumentService.getNotificationWorkflowDocumentByDocumentId(
094                    messageDelivery.getUserRecipientId(), sysId);
095    
096            flagWorkflowDocument(workflowDoc);
097    
098            notificationWorkflowDocumentService.clearAllFyisAndAcknowledgeNotificationWorkflowDocument(
099                    messageDelivery.getUserRecipientId(), workflowDoc,
100                    NotificationConstants.KEW_CONSTANTS.GENERIC_AUTO_REMOVE_ANNOTATION);
101        }
102    
103        /**
104         * @see org.kuali.rice.ken.deliverer.NotificationMessageDeliverer#dismissMessageDelivery(org.kuali.rice.ken.bo.NotificationMessageDelivery,
105         *      java.lang.String, java.lang.String)
106         */
107        public void dismissMessageDelivery(NotificationMessageDelivery messageDelivery, String user, String cause) {
108            // TODO: move hardcoded web controller actions here...
109            LOG.info("Dismissing as user '" + user + "' workflow document '" + messageDelivery.getDeliverySystemId()
110                    + "' corresponding to message delivery #" + messageDelivery.getId() + " due to cause: " + cause);
111            if (NotificationConstants.AUTO_REMOVE_CAUSE.equals(cause)) {
112                // perform an auto-remove
113                // XXX: currently auto-removes are going through autoremove method
114            } else {
115                WorkflowDocument nwd;
116                nwd = notificationWorkflowDocumentService.getNotificationWorkflowDocumentByDocumentId(user,
117                        messageDelivery.getDeliverySystemId());
118    
119                flagWorkflowDocument(nwd);
120    
121                if (NotificationConstants.ACK_CAUSE.equals(cause)) {
122                    // moved from NotificationController, ack command
123                    /*
124                     * acknowledge using workflow docId
125                     */
126                    if (nwd.isAcknowledgeRequested()) {
127                        nwd.acknowledge("This notification has been acknowledged.");
128                        LOG.debug("acknowledged " + nwd.getTitle());
129                        LOG.debug("status display value: " + nwd.getStatus().getLabel());
130                    } else {
131                        LOG.debug("Acknowledgement was not needed for document " + nwd.getDocumentId());
132                    }
133                } else if (NotificationConstants.FYI_CAUSE.equals(cause)) {
134                    // moved from NotificationController, fyi command
135                    /*
136                     * FYI using workflow docId
137                     */
138                    if (nwd.isFYIRequested()) {
139                        nwd.fyi();
140                        LOG.debug("fyi " + nwd.getTitle());
141                        LOG.debug("status display value: " + nwd.getStatus().getLabel());
142                    } else {
143                        LOG.debug("FYI was not needed for document " + nwd.getDocumentId());
144                    }
145                }
146            }
147        }
148    
149        /**
150         * Marks the workflow document as originating from the Notification System, so that the
151         * Notification post-processor does not route the action back through the Notification System.
152         * @param doc the doc to monogram
153         */
154        protected void flagWorkflowDocument(WorkflowDocument doc) {
155            Properties p = new Properties();
156            p.setProperty(INTERNAL_COMMAND_FLAG, "true");
157            ByteArrayOutputStream baos = new ByteArrayOutputStream(100);
158            try {
159                p.store(baos, null);
160            } catch (IOException ioe) {
161                throw new RuntimeException("Could not store properties", ioe);
162            }
163            doc.setAttributeContent("<whatever>" + new String(baos.toByteArray()) + "</whatever>");
164        }
165    }