Coverage Report - org.kuali.rice.kew.mail.service.impl.StyleableEmailContentServiceImpl
 
Classes in this File Line Coverage Branch Coverage Complexity
StyleableEmailContentServiceImpl
0%
0/230
0%
0/46
2.909
 
 1  
 /*
 2  
  * Copyright 2005-2007 The Kuali Foundation
 3  
  *
 4  
  *
 5  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 6  
  * you may not use this file except in compliance with the License.
 7  
  * You may obtain a copy of the License at
 8  
  *
 9  
  * http://www.opensource.org/licenses/ecl2.php
 10  
  *
 11  
  * Unless required by applicable law or agreed to in writing, software
 12  
  * distributed under the License is distributed on an "AS IS" BASIS,
 13  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 14  
  * See the License for the specific language governing permissions and
 15  
  * limitations under the License.
 16  
  */
 17  
 
 18  
 package org.kuali.rice.kew.mail.service.impl;
 19  
 
 20  
 import java.io.StringWriter;
 21  
 import java.sql.Timestamp;
 22  
 import java.util.Collection;
 23  
 import java.util.Map;
 24  
 
 25  
 import javax.xml.parsers.DocumentBuilder;
 26  
 import javax.xml.parsers.DocumentBuilderFactory;
 27  
 import javax.xml.parsers.ParserConfigurationException;
 28  
 import javax.xml.transform.Templates;
 29  
 import javax.xml.transform.TransformerConfigurationException;
 30  
 import javax.xml.transform.TransformerException;
 31  
 import javax.xml.transform.TransformerFactory;
 32  
 import javax.xml.transform.dom.DOMSource;
 33  
 import javax.xml.transform.stream.StreamResult;
 34  
 import javax.xml.transform.stream.StreamSource;
 35  
 
 36  
 import org.apache.commons.lang.StringUtils;
 37  
 import org.apache.log4j.Logger;
 38  
 import org.kuali.rice.core.api.style.StyleService;
 39  
 import org.kuali.rice.core.mail.EmailContent;
 40  
 import org.kuali.rice.core.util.RiceConstants;
 41  
 import org.kuali.rice.core.util.XmlHelper;
 42  
 import org.kuali.rice.core.util.XmlJotter;
 43  
 import org.kuali.rice.kew.actionitem.ActionItem;
 44  
 import org.kuali.rice.kew.doctype.bo.DocumentType;
 45  
 import org.kuali.rice.kew.exception.WorkflowRuntimeException;
 46  
 import org.kuali.rice.kew.feedback.web.FeedbackForm;
 47  
 import org.kuali.rice.kew.mail.CustomEmailAttribute;
 48  
 import org.kuali.rice.kew.mail.EmailStyleHelper;
 49  
 import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
 50  
 import org.kuali.rice.kew.routeheader.service.RouteHeaderService;
 51  
 import org.kuali.rice.kew.service.KEWServiceLocator;
 52  
 import org.kuali.rice.kew.user.UserUtils;
 53  
 import org.kuali.rice.kew.util.KEWConstants;
 54  
 import org.kuali.rice.kim.bo.Person;
 55  
 import org.kuali.rice.kim.bo.entity.KimPrincipal;
 56  
 import org.kuali.rice.kim.service.KIMServiceLocator;
 57  
 import org.kuali.rice.kns.util.GlobalVariables;
 58  
 import org.springframework.core.io.DefaultResourceLoader;
 59  
 import org.w3c.dom.Document;
 60  
 import org.w3c.dom.Element;
 61  
 import org.w3c.dom.Node;
 62  
 
 63  
 
 64  
 
 65  
 /**
 66  
  * EmailContentService that serves EmailContent customizable via XSLT style sheets
 67  
  * The global email style name is: kew.email.style
 68  
  * If this style is not found, the resource 'defaultEmailStyle.xsl' will be retrieved
 69  
  * relative to this class.
 70  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 71  
  */
 72  0
 public class StyleableEmailContentServiceImpl extends BaseEmailContentServiceImpl {
 73  0
     private static final Logger LOG = Logger.getLogger(StyleableEmailContentServiceImpl.class);
 74  
 
 75  0
     protected final String DEFAULT_EMAIL_STYLESHEET_RESOURCE_LOC = "defaultEmailStyle.xsl";
 76  
 
 77  
     protected StyleService styleService;
 78  0
     protected EmailStyleHelper styleHelper = new EmailStyleHelper();
 79  0
     protected String globalEmailStyleSheet = KEWConstants.EMAIL_STYLESHEET_NAME;
 80  
 
 81  
     protected RouteHeaderService routeHeaderService;
 82  
 
 83  
     public void setStyleService(StyleService styleService) {
 84  0
         this.styleService = styleService;
 85  0
     }
 86  
 
 87  
     public void setGlobalEmailStyleSheet(String globalEmailStyleSheet) {
 88  0
         this.globalEmailStyleSheet = globalEmailStyleSheet;
 89  0
     }
 90  
 
 91  
     protected static DocumentBuilder getDocumentBuilder(boolean coalesce) {
 92  
         try {
 93  0
             DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
 94  0
             dbf.setCoalescing(coalesce);
 95  0
             return dbf.newDocumentBuilder();
 96  0
         } catch (ParserConfigurationException e) {
 97  0
             String message = "Error constructing document builder";
 98  0
             LOG.error(message, e);
 99  0
             throw new WorkflowRuntimeException(message, e);
 100  
         }
 101  
     }
 102  
 
 103  
     protected static void addObjectXML(Document doc, Object o, Node node, String name) throws Exception {
 104  0
         Element element = XmlHelper.propertiesToXml(doc, o, name);
 105  
 
 106  0
         if (LOG.isDebugEnabled()) {
 107  0
             LOG.debug(XmlJotter.jotNode(element));
 108  
         }
 109  
 
 110  0
         if (node == null) {
 111  0
             node = doc;
 112  
         }
 113  
 
 114  0
         node.appendChild(element);
 115  0
     }
 116  
 
 117  
     protected static void addTextElement(Document doc, Element baseElement, String elementName, Object elementData) {
 118  0
         Element element = doc.createElement(elementName);
 119  0
         String dataValue = "";
 120  0
         if (elementData != null) {
 121  0
                 dataValue = elementData.toString();
 122  
         }
 123  0
         element.appendChild(doc.createTextNode(dataValue));
 124  0
         baseElement.appendChild(element);
 125  0
     }
 126  
 
 127  
     protected static void addCDataElement(Document doc, Element baseElement, String elementName, Object elementData) {
 128  0
         Element element = doc.createElement(elementName);
 129  0
         String dataValue = "";
 130  0
         if (elementData != null) {
 131  0
             dataValue = elementData.toString();
 132  
         }
 133  0
         element.appendChild(doc.createCDATASection(dataValue));
 134  0
         baseElement.appendChild(element);
 135  0
     }
 136  
 
 137  
     protected static void addTimestampElement(Document doc, Element baseElement, String elementName, Timestamp elementData) {
 138  0
         addTextElement(doc, baseElement, elementName, RiceConstants.getDefaultDateFormat().format(elementData));
 139  0
     }
 140  
 
 141  
     protected static void addDelegatorElement(Document doc, Element baseElement, ActionItem actionItem) {
 142  0
         Element delegatorElement = doc.createElement("delegator");
 143  0
         if ( (actionItem.getDelegatorWorkflowId() != null) && (actionItem.getDelegatorWorkflowId() != null) ) {
 144  
             // add empty delegator element
 145  0
             baseElement.appendChild(delegatorElement);
 146  0
             return;
 147  
         }
 148  0
         String delegatorType = "";
 149  0
         String delegatorId = "";
 150  0
         String delegatorDisplayValue = "";
 151  0
         if (actionItem.getDelegatorWorkflowId() != null) {
 152  0
             delegatorType = "user";
 153  0
             delegatorId = actionItem.getDelegatorWorkflowId();
 154  0
             KimPrincipal delegator = KIMServiceLocator.getIdentityManagementService().getPrincipal(delegatorId);
 155  
             
 156  0
             if (delegator == null) {
 157  0
                     LOG.error("Cannot find user for id " + delegatorId);
 158  0
                     delegatorDisplayValue = "USER NOT FOUND";
 159  
             } else {
 160  0
                     delegatorDisplayValue = UserUtils.getTransposedName(GlobalVariables.getUserSession(), delegator);
 161  
             }
 162  0
         } else if (actionItem.getDelegatorWorkflowId() != null) {
 163  0
             delegatorType = "workgroup";
 164  0
             delegatorId = actionItem.getDelegatorGroupId().toString();
 165  0
             delegatorDisplayValue = KIMServiceLocator.getIdentityManagementService().getGroup(actionItem.getDelegatorGroupId()).getGroupName();
 166  
         }
 167  0
         delegatorElement.setAttribute("type", delegatorType);
 168  
         // add the id element
 169  0
         Element idElement = doc.createElement("id");
 170  0
         idElement.appendChild(doc.createTextNode(delegatorId));
 171  0
         delegatorElement.appendChild(idElement);
 172  
         // add the display value element
 173  0
         Element displayValElement = doc.createElement("displayValue");
 174  0
         displayValElement.appendChild(doc.createTextNode(delegatorDisplayValue));
 175  0
         delegatorElement.appendChild(displayValElement);
 176  0
         baseElement.appendChild(delegatorElement);
 177  0
     }
 178  
 
 179  
     protected static void addWorkgroupRequestElement(Document doc, Element baseElement, ActionItem actionItem) {
 180  0
         Element workgroupElement = doc.createElement("workgroupRequest");
 181  0
         if (actionItem.isWorkgroupItem()) {
 182  
             // add the id element
 183  0
             Element idElement = doc.createElement("id");
 184  0
             idElement.appendChild(doc.createTextNode(actionItem.getGroupId()));
 185  0
             workgroupElement.appendChild(idElement);
 186  
             // add the display value element
 187  0
             Element displayValElement = doc.createElement("displayValue");
 188  0
             displayValElement.appendChild(doc.createTextNode(actionItem.getGroupId()));
 189  0
             workgroupElement.appendChild(displayValElement);
 190  
         }
 191  0
         baseElement.appendChild(workgroupElement);
 192  0
     }
 193  
 
 194  
     /**
 195  
      * This method is used to add the given {@link ActionItem} to the given {@link org.w3c.dom.Document} in a summarized
 196  
      * form for use in weekly or daily type reminder e-mails.
 197  
      *
 198  
      * @param doc - Document to have the ActionItem added to
 199  
      * @param actionItem - the action item being added
 200  
      * @param user - the current user
 201  
      * @param node - the node object to add the actionItem XML to (defaults to the doc variable if null is passed in)
 202  
      * @throws Exception
 203  
      */
 204  
     protected void addSummarizedActionItem(Document doc, ActionItem actionItem, Person user, Node node, DocumentRouteHeaderValue routeHeader) throws Exception {
 205  0
         if (node == null) {
 206  0
             node = doc;
 207  
         }
 208  
 
 209  0
         Element root = doc.createElement("summarizedActionItem");
 210  
 
 211  
         // add in all items from action list as preliminary default dataset
 212  0
         addTextElement(doc, root, "routeHeaderId", actionItem.getRouteHeaderId());
 213  0
         addTextElement(doc, root, "docName", actionItem.getDocName());
 214  0
         addCDataElement(doc, root, "docLabel", actionItem.getDocLabel());
 215  0
         addCDataElement(doc, root, "docTitle", actionItem.getDocTitle());
 216  
         //DocumentRouteHeaderValue routeHeader = getRouteHeader(actionItem);
 217  0
         addTextElement(doc, root, "docRouteStatus", routeHeader.getDocRouteStatus());
 218  0
         addCDataElement(doc, root, "routeStatusLabel", routeHeader.getRouteStatusLabel());
 219  0
         addTextElement(doc, root, "actionRequestCd", actionItem.getActionRequestCd());
 220  0
         addTextElement(doc, root, "actionRequestLabel", actionItem.getActionRequestLabel());
 221  0
         addDelegatorElement(doc, root, actionItem);
 222  0
         addTimestampElement(doc, root, "createDate", routeHeader.getCreateDate());
 223  0
         addWorkgroupRequestElement(doc, root, actionItem);
 224  0
         addTimestampElement(doc, root, "dateAssigned", actionItem.getDateAssigned());
 225  
 
 226  0
         node.appendChild(root);
 227  0
     }
 228  
 
 229  
     public DocumentRouteHeaderValue getRouteHeader(ActionItem actionItem) {
 230  0
             if (routeHeaderService == null) {
 231  0
                     routeHeaderService = KEWServiceLocator.getRouteHeaderService();
 232  
             }
 233  0
         return routeHeaderService.getRouteHeader(actionItem.getRouteHeaderId());
 234  
     }
 235  
 
 236  
     protected Map<Long,DocumentRouteHeaderValue> getRouteHeaders(Collection<ActionItem> actionItems) {
 237  0
             if (routeHeaderService == null) {
 238  0
                     routeHeaderService = KEWServiceLocator.getRouteHeaderService();
 239  
             }
 240  0
             return routeHeaderService.getRouteHeadersForActionItems(actionItems);
 241  
     }
 242  
     
 243  
     protected static String transform(Templates style, Document doc) {
 244  0
         StringWriter writer = new StringWriter();
 245  0
         StreamResult result = new StreamResult(writer);
 246  
 
 247  
         try {
 248  0
             style.newTransformer().transform(new DOMSource(doc), result);
 249  0
             return writer.toString();
 250  0
         } catch (TransformerException te) {
 251  0
             String message = "Error transforming DOM";
 252  0
             LOG.error(message, te);
 253  0
             throw new WorkflowRuntimeException(message, te);
 254  
         }
 255  
     }
 256  
 
 257  
     /**
 258  
      * This method retrieves the style from the system using the given name. If none is found the default style xsl file
 259  
      * defined by {@link #DEFAULT_EMAIL_STYLESHEET_RESOURCE_LOC} is used.
 260  
      *
 261  
      * @param styleName
 262  
      * @return a valid {@link javax.xml.transform.Templates} using either the given styleName or the default xsl style file
 263  
      */
 264  
     protected Templates getStyle(String styleName) {
 265  0
         Templates style = null;
 266  
         try {
 267  0
             style = styleService.getStyleAsTranslet(styleName);
 268  0
         } catch (TransformerConfigurationException tce) {
 269  0
             String message = "Error obtaining style '" + styleName + "', using default";
 270  0
             LOG.error(message, tce);
 271  
             // throw new WorkflowRuntimeException("Error obtaining style '" + styleName + "'", tce);
 272  0
         }
 273  
 
 274  0
         if (style == null) {
 275  0
             LOG.warn("Could not find specified style, " + styleName + ", using default");
 276  
             try {
 277  
 
 278  0
                 style = TransformerFactory.newInstance().newTemplates(new StreamSource(new DefaultResourceLoader().getResource("classpath:org/kuali/rice/kew/mail/" + DEFAULT_EMAIL_STYLESHEET_RESOURCE_LOC).getInputStream()));
 279  0
             } catch (Exception tce) {
 280  0
                 String message = "Error obtaining default style from resource: " + DEFAULT_EMAIL_STYLESHEET_RESOURCE_LOC;
 281  0
                 LOG.error(message, tce);
 282  0
                 throw new WorkflowRuntimeException("Error obtaining style '" + styleName + "'", tce);
 283  0
             }
 284  
         }
 285  0
         return style;
 286  
     }
 287  
 
 288  
     protected EmailContent generateEmailContent(String styleName, Document doc) {
 289  0
         Templates style = getStyle(styleName);
 290  0
         return styleHelper.generateEmailContent(style, doc);
 291  
     }
 292  
 
 293  
     protected EmailContent generateReminderForActionItems(Person user, Collection<ActionItem> actionItems, String name, String style) {
 294  0
         DocumentBuilder db = getDocumentBuilder(false);
 295  0
         Document doc = db.newDocument();
 296  0
         Element element = doc.createElement(name);
 297  0
         Map<Long,DocumentRouteHeaderValue> routeHeaders = getRouteHeaders(actionItems);
 298  
         
 299  0
         setStandardAttributes(element);
 300  0
         doc.appendChild(element);
 301  
 
 302  
         try {
 303  0
             addObjectXML(doc, user, element, "user");
 304  0
             for (ActionItem actionItem: actionItems) {
 305  
                 try {
 306  0
                     addSummarizedActionItem(doc, actionItem, user, element, routeHeaders.get(actionItem.getRouteHeaderId()));
 307  0
                 } catch (Exception e) {
 308  0
                     String message = "Error generating XML for action item: " + actionItem;
 309  0
                     LOG.error(message, e);
 310  0
                     throw new WorkflowRuntimeException(e);
 311  0
                 }
 312  
             }
 313  
 
 314  0
         } catch (Exception e) {
 315  0
             String message = "Error generating XML for action items: " + actionItems;
 316  0
             LOG.error(message, e);
 317  0
             throw new WorkflowRuntimeException(e);
 318  0
         }
 319  
 
 320  0
         return generateEmailContent(style, doc);
 321  
     }
 322  
 
 323  
     protected void setStandardAttributes(Element e) {
 324  0
         e.setAttribute("env", getDeploymentEnvironment());
 325  0
         e.setAttribute("applicationEmailAddress", getApplicationEmailAddress());
 326  0
         e.setAttribute("actionListUrl", getActionListUrl());
 327  0
         e.setAttribute("preferencesUrl", getPreferencesUrl());
 328  0
     }
 329  
 
 330  
     /**
 331  
      * This method generates an {@link EmailContent} object using the given parameters.  Part of this operation includes
 332  
      * serializing the given {@link ActionItem} to XML. The following objects and methods are included in the serialization:
 333  
      *
 334  
      * <ul>
 335  
      * <li>{@link Person}</li>
 336  
      * <li>{@link Person#getPrincipalName()}</li>
 337  
      * <li>{@link DocumentRouteHeaderValue}</li>
 338  
      * <li>{@link DocumentRouteHeaderValue#getInitiatorUser()}</li>
 339  
      * <li>{@link DocumentRouteHeaderValue#getDocumentType()}</li>
 340  
      * <li>{@link Person}</li>
 341  
      * </ul>
 342  
      *
 343  
      * @param user - the current user
 344  
      * @param actionItem - the action item being added
 345  
      * @param documentType - the document type that the custom email style sheet will come from
 346  
      * @param node - the node object to add the actionItem XML to (defaults to the doc variable if null is passed in)
 347  
      * @throws Exception
 348  
      */
 349  
     @Override
 350  
         public EmailContent generateImmediateReminder(Person user, ActionItem actionItem, DocumentType documentType) {
 351  
             
 352  0
             LOG.info("Starting generation of immediate email reminder...");
 353  0
             LOG.info("Action Id: " + actionItem.getActionItemId() + 
 354  
                              ";  ActionRequestId: " + actionItem.getActionRequestId() + 
 355  
                              ";  Action Item Principal Name: " + actionItem.getPerson().getPrincipalName());
 356  0
             LOG.info("User Principal Name: " + user.getPrincipalName());
 357  
         // change style name based on documentType when configurable email style on document is implemented...
 358  0
         String styleSheet = documentType.getCustomEmailStylesheet();
 359  0
         LOG.debug(documentType.getName() + " style: " + styleSheet);
 360  0
         if (styleSheet == null) {
 361  0
             styleSheet = globalEmailStyleSheet;
 362  
         }
 363  
 
 364  0
         LOG.info("generateImmediateReminder using style sheet: "+ styleSheet + " for Document Type " + documentType.getName());
 365  
 //        return generateReminderForActionItems(user, actionItems, "immediateReminder", styleSheet);
 366  0
         DocumentBuilder db = getDocumentBuilder(false);
 367  0
         Document doc = db.newDocument();
 368  0
         Element element = doc.createElement("immediateReminder");
 369  0
         setStandardAttributes(element);
 370  0
         doc.appendChild(element);
 371  
 
 372  
         try {
 373  0
             addObjectXML(doc, user, element, "user");
 374  
 //            addActionItem(doc, actionItem, user, node);
 375  0
             Node node = element;
 376  0
             if (node == null) {
 377  0
                 node = doc;
 378  
             }
 379  
 
 380  0
             Element root = doc.createElement("actionItem");
 381  
             // append the custom body and subject if they exist
 382  
             try {
 383  0
                 CustomEmailAttribute customEmailAttribute = getCustomEmailAttribute(user, actionItem);
 384  0
                 if (customEmailAttribute != null) {
 385  0
                     String customBody = customEmailAttribute.getCustomEmailBody();
 386  0
                     if (!org.apache.commons.lang.StringUtils.isEmpty(customBody)) {
 387  0
                         Element bodyElement = doc.createElement("customBody");
 388  0
                         bodyElement.appendChild(doc.createTextNode(customBody));
 389  0
                         root.appendChild(bodyElement);
 390  
                     }
 391  0
                     String customEmailSubject = customEmailAttribute.getCustomEmailSubject();
 392  0
                     if (!org.apache.commons.lang.StringUtils.isEmpty(customEmailSubject)) {
 393  0
                         Element subjectElement = doc.createElement("customSubject");
 394  0
                         subjectElement.appendChild(doc.createTextNode(customEmailSubject));
 395  0
                         root.appendChild(subjectElement);
 396  
                     }
 397  
                 }
 398  0
             } catch (Exception e) {
 399  0
                 LOG.error("Error when checking for custom email body and subject.", e);
 400  0
             }
 401  0
             Person person = actionItem.getPerson();
 402  0
             DocumentRouteHeaderValue header = getRouteHeader(actionItem);
 403  
             // keep adding stuff until we have all the xml we need to formulate the message :/
 404  0
             addObjectXML(doc, actionItem, root, "actionItem");
 405  0
             addObjectXML(doc, person, root, "actionItemPerson");
 406  0
             addTextElement(doc, root, "actionItemPrincipalId", person.getPrincipalId());
 407  0
             addTextElement(doc, root, "actionItemPrincipalName", person.getPrincipalName());
 408  0
             addDocumentHeaderXML(doc, header, root, "doc");
 409  0
             addObjectXML(doc, header.getInitiatorPrincipal(), root, "docInitiator");
 410  0
             addTextElement(doc, root, "docInitiatorDisplayName", header.getInitiatorDisplayName());
 411  0
             addObjectXML(doc, header.getDocumentType(), root, "documentType");
 412  
 
 413  0
             node.appendChild(root);
 414  0
         } catch (Exception e) {
 415  0
             String message = "Error generating immediate reminder XML for action item: " + actionItem;
 416  0
             LOG.error(message, e);
 417  0
             throw new WorkflowRuntimeException(e);
 418  0
         }
 419  0
         LOG.info("Leaving generation of immeidate email reminder...");
 420  
             /**
 421  
              * End IU customization
 422  
              */
 423  0
         return generateEmailContent(styleSheet, doc);
 424  
     }
 425  
     
 426  
     /**
 427  
      * This method handles converting the DocumentRouteHeaderValue into an XML representation.  The reason we can't just use
 428  
      * propertiesToXml like we have elsewhere is because the doc header has a String attached to it that has the XML document
 429  
      * content in it.  The default serialization of this will serialize this as a String so we will end up with escaped XML
 430  
      * in our output which we won't be able to process with the email stylesheet.  So we need to read the xml content from
 431  
      * the document and parse it into a DOM object so it can be appended to our output.
 432  
      */
 433  
     protected void addDocumentHeaderXML(Document document, DocumentRouteHeaderValue documentHeader, Node node, String elementName) throws Exception {
 434  0
             Element element = XmlHelper.propertiesToXml(document, documentHeader, elementName);
 435  
             // now we need to "fix" the xml document content because it's going to be in there as escaped XML
 436  0
             Element docContentElement = (Element)element.getElementsByTagName("docContent").item(0);
 437  0
             String documentContent = docContentElement.getTextContent();
 438  
             
 439  0
             if (!StringUtils.isBlank(documentContent) && documentContent.startsWith("<")) {
 440  0
                     Document documentContentXML = XmlHelper.readXml(documentContent);
 441  0
                     Element documentContentElement = documentContentXML.getDocumentElement();
 442  0
                     documentContentElement = (Element)document.importNode(documentContentElement, true);
 443  
             
 444  
                     // remove the old, bad text content
 445  0
                     docContentElement.removeChild(docContentElement.getFirstChild());
 446  
             
 447  
                     // replace with actual XML
 448  0
                     docContentElement.appendChild(documentContentElement);
 449  0
             } else {
 450  
                     // in this case it means that the XML is encrypted, unfortunately, we have no way to decrypt it since
 451  
                     // the key is stored in the client application.  We will just include the doc content since none of our
 452  
                     // current IU clients will be using this feature right away
 453  
 
 454  
                     // remove the old, bad text content
 455  0
                     docContentElement.removeChild(docContentElement.getFirstChild());
 456  
             }
 457  
             
 458  0
             if (LOG.isDebugEnabled()) {
 459  0
             LOG.debug(XmlJotter.jotNode(element));
 460  
         }
 461  
 
 462  0
         node.appendChild(element);
 463  0
     }
 464  
 
 465  
     @Override
 466  
         public EmailContent generateWeeklyReminder(Person user, Collection<ActionItem> actionItems) {
 467  0
         return generateReminderForActionItems(user, actionItems, "weeklyReminder", globalEmailStyleSheet);
 468  
     }
 469  
 
 470  
     @Override
 471  
         public EmailContent generateDailyReminder(Person user, Collection<ActionItem> actionItems) {
 472  0
         return generateReminderForActionItems(user, actionItems, "dailyReminder", globalEmailStyleSheet);
 473  
     }
 474  
 
 475  
     @Override
 476  
         public EmailContent generateFeedback(FeedbackForm form) {
 477  0
         DocumentBuilder db = getDocumentBuilder(true);
 478  0
         Document doc = db.newDocument();
 479  0
         String styleSheet = globalEmailStyleSheet;
 480  
 
 481  
         // if the doc type is specified, see if that doc has a custom email stylesheet and use it
 482  
         // NOTE: do we need to do this for feedback? presumably feedback will be going back to admins
 483  
         /*String docTypeName = form.getDocumentType();
 484  
         if (!StringUtils.isBlank(docTypeName)) {
 485  
             DocumentType docType = KEWServiceLocator.getDocumentTypeService().findByName(docTypeName);
 486  
             if (docType == null) {
 487  
                 LOG.error("User specified document type '" + docTypeName + "' in feedback form, but the document type was not found in the system");
 488  
             } else {
 489  
                 if (docType.getCustomEmailStylesheet() != null) {
 490  
                     styleSheet = docType.getCustomEmailStylesheet();
 491  
                 }
 492  
             }
 493  
         }*/
 494  0
         LOG.info("form: " + form.getRouteHeaderId());
 495  
         try {
 496  0
             addObjectXML(doc, form, null, "feedback");
 497  0
         } catch (Exception e) {
 498  0
             String message = "Error generating XML for feedback form: " + form;
 499  0
             LOG.error(message, e);
 500  0
             throw new WorkflowRuntimeException(message, e);
 501  0
         }
 502  0
         setStandardAttributes(doc.getDocumentElement());
 503  
 
 504  0
         return generateEmailContent(styleSheet, doc);
 505  
     }
 506  
 }