com.adobe.acs.commons.email.process.impl.SendTemplatedEmailProcess Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of acs-aem-commons-bundle Show documentation
Show all versions of acs-aem-commons-bundle Show documentation
Main ACS AEM Commons OSGi Bundle. Includes commons utilities.
/*
* ACS AEM Commons
*
* Copyright (C) 2013 - 2023 Adobe
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.adobe.acs.commons.email.process.impl;
import java.text.SimpleDateFormat;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.jcr.RepositoryException;
import org.apache.commons.lang.StringUtils;
import org.apache.felix.scr.annotations.Activate;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Deactivate;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Reference;
import org.apache.felix.scr.annotations.Service;
import org.apache.jackrabbit.api.security.user.Authorizable;
import org.apache.jackrabbit.api.security.user.UserManager;
import org.apache.sling.api.resource.LoginException;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ResourceResolverFactory;
import org.apache.sling.jcr.resource.api.JcrResourceConstants;
import org.osgi.service.component.ComponentContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.adobe.acs.commons.email.EmailService;
import com.adobe.acs.commons.wcm.AuthorUIHelper;
import com.day.cq.commons.Externalizer;
import com.day.cq.dam.commons.util.DamUtil;
import com.day.cq.workflow.WorkflowException;
import com.day.cq.workflow.WorkflowSession;
import com.day.cq.workflow.exec.WorkItem;
import com.day.cq.workflow.exec.WorkflowData;
import com.day.cq.workflow.exec.WorkflowProcess;
import com.day.cq.workflow.metadata.MetaDataMap;
/**
* This abstract SendTemplatedEmailProcess
class is a WorkFlow process step
* that will send an email using {@link com.adobe.acs.commons.email.EmailService EmailService}.
* By default jcr properties from the payload are automatically added to the email parameter
* map. If the payload is a cq:Page then the properties at the jcr:content level
* are added. If the payload is a dam:Asset then the properties at the metadata
* node are added. In addition the parameters outlined in
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailConstants
* SendTemplatedEmailConstants} are also automatically added.
* This process will send the email to a CQ user or members of a CQ group
* specified by the process argument
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailProcess.Arguments#SEND_TO
* SEND_TO} Implementing classes can override this logic by overriding the
* method:
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailProcess#getEmailAddrs(com.day.cq.workflow.exec.WorkItem, org.apache.sling.api.resource.Resource, java.lang.String[])
* getEmailAddrs() - method}
* Implementing classes can also add additional parameters by overriding the
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailProcess#getAdditionalParams(WorkItem, WorkflowSession, Resource)
* getAdditionalParams() - method}
*
*
*
* Process Configuration This process supports the following
* configuration arguments:
*
* - emailTemplate
* - String representing a path to the template to be used to send the email.
* If no template is set no email is sent.
* - sendTo
* - String representing a path to a user or group. If the path is a user an
* email will be sent to that user. if the path is to a group then the email
* will be sent to all members of that group.
* - dateFormat
* - An optional parameter to specify how jcr Date properties are converted to
* Strings. The format should be set using the
* {@link java.text.SimpleDateFormat java.text.SimpleDateFormat}. Defaults to
*
yyyy-MM-dd hh:mm a
*
*
*/
@Component
@Property(label = "Workflow Label", name = "process.label", value = "Send Templated Email", description = "Sends a templated email using the ACS Commons Email Service")
@Service
public class SendTemplatedEmailProcess implements WorkflowProcess {
private static final Logger log = LoggerFactory.getLogger(SendTemplatedEmailProcess.class);
/**
* Service used to send the email
*/
@Reference
private EmailService emailService;
/**
* Service used to generate a link to the payload on author environment
*/
@Reference
private AuthorUIHelper authorUiHelper;
@Reference
private ResourceResolverFactory resourceResolverFactory;
/**
* used to generate a link to the payload on publish environment
*/
@Reference
private Externalizer externalizer;
/**
* The available arguments to this process implementation.
*/
protected enum Arguments {
PROCESS_ARGS("PROCESS_ARGS"),
/**
* emailTemplate - process argument
*/
TEMPLATE("emailTemplate"),
/**
* sendTo - process argument
*/
SEND_TO("sendTo"),
/**
* dateFormat - process argument
*/
DATE_FORMAT("dateFormat");
private String argumentName;
Arguments(String argumentName) {
this.argumentName = argumentName;
}
public String getArgumentName() {
return this.argumentName;
}
}
@Override
public final void execute(WorkItem workItem, WorkflowSession workflowSession, MetaDataMap metaData)
throws WorkflowException {
final WorkflowData workflowData = workItem.getWorkflowData();
final String type = workflowData.getPayloadType();
// Check if the payload is a path in the JCR
if (!StringUtils.equals(type, "JCR_PATH")) {
return;
}
String[] args = buildArguments(metaData);
// process arguments
String emailTemplate = getValueFromArgs(Arguments.TEMPLATE.getArgumentName(), args);
if (emailTemplate == null) {
log.warn("Invalid process arguments, returning");
return;
}
// set date format to be used in emails
String sdfParam = getValueFromArgs(Arguments.DATE_FORMAT.getArgumentName(), args);
SimpleDateFormat sdf = getSimpleDateFormat(sdfParam);
// Get the path to the JCR resource from the payload
final String payloadPath = workflowData.getPayload().toString();
// Get ResourceResolver
final Map authInfo = new HashMap();
authInfo.put(JcrResourceConstants.AUTHENTICATION_INFO_SESSION, workflowSession.getSession());
try (ResourceResolver resourceResolver = resourceResolverFactory.getResourceResolver(authInfo) ) {
Resource payloadRes = resourceResolver.getResource(payloadPath);
// Email Parameter map
Map emailParams = new HashMap();
// Set jcr path
emailParams.put(SendTemplatedEmailConstants.JCR_PATH, payloadPath);
// Get Payload params
Map payloadProp = SendTemplatedEmailUtils.getPayloadProperties(payloadRes, sdf);
if (payloadProp != null) {
emailParams.putAll(payloadProp);
}
// Get Url params
Map urlParams = getUrls(payloadRes);
emailParams.putAll(urlParams);
// Get Additional Parameters to add
Map wfParams = getAdditionalParams(workItem, workflowSession, payloadRes);
emailParams.putAll(wfParams);
// get email addresses based on CQ user or group
String[] emailTo = getEmailAddrs(workItem, payloadRes, args);
List failureList = emailService.sendEmail(emailTemplate, emailParams, emailTo);
if (failureList.isEmpty()) {
log.info("Email sent successfully to {} recipients", emailTo.length);
} else {
log.error("Email sent failed");
}
} catch (LoginException e) {
log.error("Could not acquire a ResourceResolver object from the Workflow Session's JCR Session: {}", e);
}
}
/***
* Gets a String[] of email addresses to send the email to. By default calls
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailUtils#getEmailAddrsFromUserPath(ResourceResolver, String)}
* Protected so that it can be overridden by implementing classes to add
* unique logic to where emails are routed to.
*
* @param workItem
* the current WorkItem in the workflow
* @param payloadResource
* the current payload as a Resource
* @param args
* process arguments configured by the workflow step
* @return String[] of email addresses
*/
protected String[] getEmailAddrs(WorkItem workItem, Resource payloadResource, String[] args) {
ResourceResolver resolver = payloadResource.getResourceResolver();
String sendToUser = getValueFromArgs(Arguments.SEND_TO.getArgumentName(), args);
return SendTemplatedEmailUtils.getEmailAddrsFromPathOrName(resolver, sendToUser);
}
/***
* Returns a Map of additional parameters that will be added
* to the full list of email parameters that is sent to the EmailService. By
* default adds the Workflow Title:
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailConstants#WF_MODEL_TITLE
* WF_MODEL_TITLE} and adds the Workflow Step Title:
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailConstants#WF_STEP_TITLE
* WF_STEP_TITLE}
* {@link com.adobe.acs.commons.email.process.impl.SendTemplatedEmailConstants#WF_INITIATOR
* WF_INITIATOR} Protected so that implementing classes can override and
* add additional parameters.
*
* @param workItem
* @param workflowSession
* @param payloadResource
* @return Map of additional parameters to be added to email
* params
*/
protected Map getAdditionalParams(WorkItem workItem, WorkflowSession workflowSession,
Resource payloadResource) {
Map wfParams = new HashMap();
try {
wfParams.put(SendTemplatedEmailConstants.WF_STEP_TITLE, workItem.getNode().getTitle());
wfParams.put(SendTemplatedEmailConstants.WF_MODEL_TITLE, workItem.getWorkflow().getWorkflowModel()
.getTitle());
// Set workflow initiator
wfParams.put(SendTemplatedEmailConstants.WF_INITIATOR, workItem.getWorkflow().getInitiator());
if(workItem.getMetaDataMap().containsKey("comment")) {
wfParams.put(SendTemplatedEmailConstants.WF_STEP_COMMENT, workItem.getMetaDataMap().get("comment").toString());
}
} catch (Exception e) {
log.warn("Error getting workflow title and workflow step title {}", e);
}
return wfParams;
}
/***
* Gets value from workflow process arguments
*
* @param key
* @param arguments
* @return String of the argument value or null if not found
*/
protected String getValueFromArgs(String key, String[] arguments) {
for (String str : arguments) {
String trimmedStr = str.trim();
if (trimmedStr.startsWith(key + ":")) {
return trimmedStr.substring((key + ":").length());
}
}
return null;
}
/***
* Uses the AuthorUIHelper to generate links to the payload on author Uses
* Externalizer to generate links to the payload on publish
*
* @param payloadRes
* @return
*/
private Map getUrls(Resource payloadRes) {
Map urlParams = new HashMap();
if (payloadRes == null) {
return urlParams;
}
String payloadPath = payloadRes.getPath();
ResourceResolver resolver = payloadRes.getResourceResolver();
if (DamUtil.isAsset(payloadRes)) {
// add author url
String assetDetailsUrl = authorUiHelper.generateEditAssetLink(payloadPath, true, resolver);
urlParams.put(SendTemplatedEmailConstants.AUTHOR_LINK, assetDetailsUrl);
// add publish url
String publishUrl = externalizer.publishLink(resolver, payloadPath);
urlParams.put(SendTemplatedEmailConstants.PUBLISH_LINK, publishUrl);
} else {
// add absolute author url
String assetDetailsUrl = authorUiHelper.generateEditPageLink(payloadPath, true, resolver);
urlParams.put(SendTemplatedEmailConstants.AUTHOR_LINK, assetDetailsUrl);
// add publish url
String publishUrl = externalizer.publishLink(resolver, payloadPath + ".html");
urlParams.put(SendTemplatedEmailConstants.PUBLISH_LINK, publishUrl);
}
return urlParams;
}
/***
*
* @param metaData
* @return
*/
private String[] buildArguments(MetaDataMap metaData) {
// the 'old' way, ensures backward compatibility
String processArgs = metaData.get(Arguments.PROCESS_ARGS.getArgumentName(), String.class);
if (processArgs != null && !processArgs.equals("")) {
return processArgs.split(",");
} else {
return new String[0];
}
}
/***
* Set the format to be used for displaying dates in the email Defaults to
* format of 'yyyy-MM-dd hh:mm a'
*
* @param formatString
* - workflow process argument to override default format
* @return SimpleDateFormat that will be used to convert jcr Date properties
* to Strings
*/
private SimpleDateFormat getSimpleDateFormat(String formatString) {
SimpleDateFormat defaultFormat = new SimpleDateFormat("yyyy-MM-dd hh:mm a");
if (formatString == null || formatString.isEmpty()) {
return defaultFormat;
}
try {
return new SimpleDateFormat(formatString);
} catch (IllegalArgumentException e) {
// invalid pattern
return defaultFormat;
}
}
@Activate
protected void activate(ComponentContext context) throws RepositoryException {
// activate
}
@Deactivate
protected void deactivate(ComponentContext context) {
// deactivate
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy