plugins/aem/cloud-service/skills/aem-workflow/workflow-development/SKILL.md
Implement custom AEM Workflow Java components on AEM as a Cloud Service. Use when writing WorkflowProcess steps, ParticipantStepChooser implementations, registering services via OSGi DS R6 annotations, reading step arguments from MetaDataMap, accessing JCR payload via WorkflowSession adapter, reading and writing workflow metadata and variables, and handling errors with WorkflowException for retry behavior.
npx skillsauth add adobe/skills workflow-developmentInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Implement custom workflow components for AEM Cloud Service: WorkflowProcess, ParticipantStepChooser, OSGi registration, metadata handling, and error patterns.
org.osgi.service.component.annotations.*). Do not use Felix SCR.ui.apps content package and is deployed via Cloud Manager pipeline.Development Progress
- [ ] 1) Identify what the step does: process (auto) or participant (human) or dynamic participant
- [ ] 2) Create Java class implementing WorkflowProcess or ParticipantStepChooser
- [ ] 3) Register with correct @Component annotation and service property (process.label / chooser.label)
- [ ] 4) Read step arguments from MetaDataMap args (set in model editor)
- [ ] 5) Access payload via item.getWorkflowData().getPayload().toString()
- [ ] 6) Read/write workflow instance metadata via item.getWorkflowData().getMetaDataMap()
- [ ] 7) Return normally to advance; throw WorkflowException to trigger retry
- [ ] 8) Deploy bundle; verify process.label appears in Workflow Model Editor step picker
@Component(
service = WorkflowProcess.class,
property = {
"process.label=My Custom Process Step",
"service.description=Short description of what this step does"
}
)
public class MyCustomProcess implements WorkflowProcess {
private static final Logger LOG = LoggerFactory.getLogger(MyCustomProcess.class);
@Reference
private ResourceResolverFactory resolverFactory;
@Override
public void execute(WorkItem item, WorkflowSession session, MetaDataMap args)
throws WorkflowException {
// 1. Read payload
WorkflowData data = item.getWorkflowData();
String payloadPath = data.getPayload().toString();
// 2. Read step arguments
String recipient = args.get("recipient", "workflow-administrators");
boolean createVersion = args.get("createVersion", false);
// 3. Read/write shared workflow metadata
MetaDataMap metadata = data.getMetaDataMap();
String status = metadata.get("approvalStatus", "PENDING");
metadata.put("processedBy", "my-custom-step");
// 4. Access JCR if needed
try {
ResourceResolver resolver = session.adaptTo(ResourceResolver.class);
Resource resource = resolver.getResource(payloadPath);
// ... do work ...
} catch (Exception e) {
LOG.error("Error in MyCustomProcess for payload {}", payloadPath, e);
throw new WorkflowException("Failed: " + e.getMessage(), e);
}
// Return normally = step completes, workflow advances
}
}
@Component(
service = ParticipantStepChooser.class,
property = {"chooser.label=Content Owner Chooser"}
)
public class ContentOwnerChooser implements ParticipantStepChooser {
@Override
public String getParticipant(WorkItem workItem, WorkflowSession session,
MetaDataMap args) throws WorkflowException {
String payloadPath = workItem.getWorkflowData().getPayload().toString();
try {
Session jcrSession = session.adaptTo(Session.class);
Node content = jcrSession.getNode(payloadPath + "/jcr:content");
if (content.hasProperty("cq:lastModifiedBy")) {
return content.getProperty("cq:lastModifiedBy").getString();
}
} catch (RepositoryException e) {
throw new WorkflowException("Cannot resolve participant", e);
}
return args.get("fallbackParticipant", "content-authors");
}
}
ResourceResolverFactory.loginAdministrative(). Always use a service user sub-service.Session.save() on the workflow session's JCR session for payload changes — use a separate ResourceResolver obtained from resolverFactory.getServiceResourceResolver().PROCESS_AUTO_ADVANCE=false in the model metaData and use TaskWorkflowProcess or an external completion mechanism.tools
Identifies which items (pages, campaigns, products, channels, regions) had the biggest increases or decreases for a key metric between two time periods. Use this skill when someone asks "what's up and what's down," "which campaigns moved the most," "top gainers and losers," "what pages are trending," "show me what changed by channel," or any variation of identifying the biggest movers and decliners for a metric.
tools
Compares the performance of two or more audience segments across key metrics side by side. Use this skill when someone wants to compare audiences, cohorts, or groups — for example, "how do mobile users compare to desktop users on conversion," "compare new vs. returning visitors," "show me the difference between these two segments," "compare these audiences on our KPIs," or "which segment performs better." Also trigger for "segment comparison," "audience comparison," or "cohort comparison."
business
Produces a compact KPI digest showing how key metrics changed over a period and what's driving the movement. Use this skill when someone asks for a performance summary, a weekly recap, a morning briefing, a KPI update, or any variation of "how did we do this week/month." Also trigger for requests like "give me a performance overview," "what moved in the last 7 days," "pull our KPI report," or "summarize our metrics."
testing
Analyzes a multi-step conversion funnel to find where users drop off and which steps have the worst leakage. Use this skill when someone describes a journey or funnel and asks about conversion rates, drop-off, fallout, or step completion. Trigger for phrases like "analyze our onboarding funnel," "where are users dropping off," "what's our checkout conversion rate," "funnel analysis," "show me fallout between these steps," or "which step loses the most users."