Extension des fonctionnalités de workflows extending-workflow-functionality
Cette rubrique décrit le développement de composants d’étape personnalisés pour vos workflows. Elle explique également comment interagir par programmation avec les workflows.
La création d’une étape de workflow personnalisée implique les activités suivantes :
- Développement du composant d’étape de workflow.
- Mise en œuvre de la fonctionnalité d’étape en tant que service OSGi ou script ECMA.
Vous pouvez également interagir avec les workflows de vos programmes et scripts.
Composants d’étape de workflow - Principes de base workflow-step-components-the-basics
Un composant d’étape de workflow définit l’apparence et le comportement de l’étape lors de la création de modèles de workflow :
- Nom de catégorie et d’étape dans le sidekick du workflow.
- Apparence de l’étape dans les modèles de workflow.
- Boîte de dialogue de modification pour la configuration des propriétés du composant.
- Service ou script exécuté au moment de l’exécution.
Comme avec tous les composants, les composants d’étape de workflow héritent du composant indiqué pour la propriété sling:resourceSuperType
. Le diagramme suivant présente la hiérarchie des nœuds cq:component
qui constituent la base de tous les composants des étapes de workflow. Le diagramme inclut également les composants Étape du processus, Étape du participant et Étape du participant dynamique, car il s’agit des points de départ les plus courants (et les plus simples) pour développer des composants d’étape personnalisée.
/libs
./libs
est remplacé dès que vous mettez à niveau votre instance (et risque de l’être si vous appliquez un correctif ou un Feature Pack).- Recréez l’élément requis (tel qu’il existe dans
/libs
sous/apps
). - Apportez les modifications désirées dans
/apps
.
Le composant /libs/cq/workflow/components/model/step
est l’ancêtre commun le plus proche de l’Étape du processus, l’Étape du participant et l’Étape du participant dynamique, qui héritent tous des éléments suivants :
-
step.jsp
Le script
step.jsp
affiche le titre du composant d’étape lorsqu’il est ajouté à un modèle. -
Boîte de dialogue avec les onglets suivants :
- Courant : pour modifier le titre et la description.
- Avancé : pour modifier les propriétés des notifications par e-mail.
note note NOTE Lorsque les onglets de la boîte de dialogue de modification d’un composant d’étape ne correspondent pas à cette apparence par défaut, le composant d’étape propose des scripts, des propriétés de nœud ou des onglets de boîte de dialogue définis qui remplacent ces onglets hérités.
Scripts ECMA ecma-scripts
Les objets suivants sont disponibles (en fonction du type d’étape) dans les scripts ECMA :
-
WorkItem workItem
-
WorkflowSession workflowSession
-
WorkflowData workflowData
-
args
: tableau contenant les arguments du processus. -
sling
: pour accéder à d’autres services osgi. -
jcrSession
MetaDataMaps metadatamaps
Vous pouvez utiliser les métadonnées de processus pour conserver les informations requises pendant la durée de vie du processus. Les étapes d’un workflow sont généralement nécessaires pour conserver les données en vue d’une utilisation ultérieure dans le workflow ou pour récupérer les données persistantes.
Il existe trois types d’objets MetaDataMap pour les objets Workflow
, WorkflowData
et WorkItem
. Ils ont tous la même fonction : stocker les métadonnées.
Un objet WorkItem possède son propre objet MetaDataMap qui peut être utilisé uniquement pendant l’exécution de cet élément de travail (l’étape en question).
Les objets MetaDataMap Workflow
et WorkflowData
sont partagés sur l’ensemble du workflow. Pour ces cas, il est recommandé d’utiliser uniquement l’objet MetaDataMap WorkflowData
.
Création de composants d’étape de workflow personnalisés creating-custom-workflow-step-components
Les composants d’étape de workflow peuvent être créés de la même manière que tout autre composant.
Pour hériter de l’un des composants de l’étape de base (existante), ajoutez la propriété suivante au nœud cq:Component
:
-
Nom :
sling:resourceSuperType
-
Type :
String
-
Valeur : l’un des chemins suivants qui se résout en un composant de base :
cq/workflow/components/model/process
cq/workflow/components/model/participant
cq/workflow/components/model/dynamic_participant
Spécification du titre et de la description par défaut pour les instances d’étape specifying-the-default-title-and-description-for-step-instances
Procédez comme suit pour spécifier les valeurs par défaut des champs Titre et Description dans l’onglet Courant.
-
La boîte de dialogue de modification de l’étape stocke le titre et la description dans les emplacements suivants : >
-
./jcr:title
-
Emplacements
./jcr:description
Cette exigence est satisfaite lorsque la boîte de dialogue de modification utilise l’onglet Commun que le composant
/libs/cq/flow/components/step/step
implémente. -
Le composant d’étape ou un ancêtre du composant ne remplace pas le script
step.jsp
que le composant/libs/cq/flow/components/step/step
implémente.
-
Sous le nœud
cq:Component
, ajoutez le nœud suivant :- Nom :
cq:editConfig
- Type :
cq:EditConfig
note note NOTE Pour plus d’informations sur le nœud cq:editConfig, consultez Configuration du comportement de modification d’un composant. - Nom :
-
Sous le nœud
cq:EditConfig
, ajoutez le nœud suivant :- Nom :
cq:formParameters
- Type :
nt:unstructured
- Nom :
-
Ajoutez les propriétés
String
des noms suivants au nœudcq:formParameters
:jcr:title
: la valeur remplit le champ Titre de l’onglet Commun.jcr:description
: la valeur remplit le champ Description de l’onglet Commun.
Enregistrement des valeurs de propriété dans les métadonnées de workflow saving-property-values-in-workflow-metadata
La propriété de nom des éléments cq:Widget
spécifie le nœud JCR qui stocke la valeur du widget. Lorsque des widgets dans la boîte de dialogue de composant de l’étape du workflow stockent des valeurs sous la balise ./metaData
, la valeur est ajoutée au workflow MetaDataMap
.
Par exemple, un champ de texte dans une boîte de dialogue est un nœud cq:Widget
qui possède les propriétés suivantes :
xtype
String
textarea
name
String
./metaData/subject
fieldLabel
String
Email Subject
La valeur spécifiée dans ce champ de texte est ajoutée à l’objet [MetaDataMap](#metadatamaps)
de l’instance de workflow et est associée à la clé subject
.
PROCESS_ARGS
, la valeur est immédiatement disponible dans les implémentations de script ECMA via la variable args
. Dans ce cas, la valeur de la propriété name est ./metaData/PROCESS_ARGS.
Remplacement de la mise en œuvre de l’étape overriding-the-step-implementation
Chaque composant d’étape de base permet à l’équipe de développement des modèles de workflow de configurer les fonctionnalités clés suivantes lors de la conception :
- Étape du processus : service ou script ECMA à exécuter au moment de l’exécution.
- Participant : ID de l’utilisateur auquel est affecté l’élément de travail généré.
- Étape du participant dynamique : le service ou le script ECMA qui sélectionne l’identifiant de l’utilisateur ou l’utilisatrice auquel est affecté l’élément de travail.
Pour cibler le composant en vue de l’utiliser dans un scénario de workflow spécifique, configurez la fonctionnalité clé dans la conception et supprimez la possibilité pour les équipes de développement de modèles de la modifier.
-
Sous le nœud cq:component, ajoutez le nœud suivant :
- Nom :
cq:editConfig
- Type :
cq:EditConfig
Pour plus d’informations sur le nœud cq:editConfig, consultez Configuration du comportement de modification d’un composant.
- Nom :
-
Sous le nœud cq:EditConfig, ajoutez le nœud suivant :
- Nom :
cq:formParameters
- Type :
nt:unstructured
- Nom :
-
Ajoutez une propriété
String
au nœudcq:formParameters
. Le supertype de composant détermine le nom de la propriété :- Étape du processus :
PROCESS
- Étape du participant :
PARTICIPANT
- Étape de participant dynamique :
DYNAMIC_PARTICIPANT
- Étape du processus :
-
Définissez la valeur de la propriété :
PROCESS
: chemin d’accès au script ECMA ou au PID du service qui implémente le comportement de l’étape.PARTICIPANT
: ID de l’utilisateur à qui l’élément de travail a été affecté.DYNAMIC_PARTICIPANT
: chemin d’accès au script ECMA ou au PID du service qui sélectionne l’utilisateur auquel affecter l’élément de travail.
-
Pour empêcher l’équipe de développement des modèles de modifier vos valeurs de propriété, remplacez la boîte de dialogue du supertype de composant.
Ajout de formulaires et de boîtes de dialogue aux étapes du participant adding-forms-and-dialogs-to-participant-steps
Personnalisez votre composant d’étape de participant pour proposer des fonctionnalités disponibles avec les composants Étape de participant du formulaire et Étape de participant de la boîte de dialogue :
- Présentez un formulaire à l’utilisateur lorsqu’il ouvre l’élément de travail généré.
- Présentez une boîte de dialogue personnalisée à l’utilisateur lorsqu’il effectue l’élément de travail généré.
Effectuez la procédure suivante sur votre nouveau composant (voir Création de composants d’étape de workflow personnalisée) :
-
Sous le nœud
cq:Component
, ajoutez le nœud suivant :- Nom :
cq:editConfig
- Type :
cq:EditConfig
Pour plus d’informations sur le nœud cq:editConfig, consultez Configuration du comportement de modification d’un composant.
- Nom :
-
Sous le nœud cq:EditConfig, ajoutez le nœud suivant :
- Nom :
cq:formParameters
- Type :
nt:unstructured
- Nom :
-
Pour présenter un formulaire lorsque l’utilisateur ouvre l’élément de travail, ajoutez la propriété suivante au nœud
cq:formParameters
:- Nom :
FORM_PATH
- Type :
String
- Valeur : chemin d’accès qui résout le formulaire
- Nom :
-
Pour présenter une boîte de dialogue personnalisée lorsque l’utilisateur effectue l’élément de travail, ajoutez la propriété suivante au nœud
cq:formParameters
.- Nom :
DIALOG_PATH
- Type :
String
- Valeur : chemin d’accès qui résout la boîte de dialogue
- Nom :
Configuration du comportement d’exécution de l’étape de workflow configuring-the-workflow-step-runtime-behavior
Sous le nœud cq:Component
, ajoutez un nœud cq:EditConfig
. En dessous, ajoutez un nœud nt:unstructured
(doit être nommé cq:formParameters
) et ajoutez à ce nœud les propriétés suivantes :
-
Nom :
PROCESS_AUTO_ADVANCE
-
Type :
Boolean
-
Valeur :
- Lorsque la propriété est définie sur
true
, le workflow exécute cette étape et se poursuit (c’est le paramètre par défaut qui est également recommandé). - Si sa valeur est
false
, le processus s’exécute et s’arrête. Ceci nécessite une manipulation supplémentaire, donc la valeurtrue
est recommandée.
- Lorsque la propriété est définie sur
-
-
Nom :
DO_NOTIFY
- Type :
Boolean
- Valeur : indique si des notifications par e-mail doivent être envoyées pour les étapes de participation de la personne (et suppose que le serveur de messagerie est correctement configuré).
- Type :
Persistance et accès aux données persisting-and-accessing-data
Données persistantes pour les étapes de workflow suivantes persisting-data-for-subsequent-workflow-steps
Vous pouvez utiliser les métadonnées de processus pour conserver les informations requises pendant la durée de vie du processus et entre les étapes. Les étapes d’un workflow sont généralement nécessaires pour conserver les données en vue d’une utilisation ultérieure ou pour récupérer les données persistantes des étapes précédentes.
Les métadonnées de processus sont stockées dans un objet MetaDataMap
. L’API Java fournit la méthode Workflow.getWorkflowData
pour renvoyer un objet WorkflowData
qui fournit l’objet MetaDataMap
approprié. Cet objet WorkflowData
MetaDataMap
est disponible pour le service OSGi ou le script ECMA d’un composant d’étape.
Java java
La méthode d’exécution de l’implémentation WorkflowProcess
est transmise à l’objet WorkItem
. Utilisez cet objet afin d’obtenir l’objet WorkflowData
pour l’instance de workflow active. L’exemple suivant ajoute un élément à l’objet workflow MetaDataMap
, puis enregistre chaque élément. L’élément ("mykey", "My Step Value") est disponible pour les étapes suivantes du workflow.
public void execute(WorkItem item, WorkflowSession session, MetaDataMap args) throws WorkflowException {
MetaDataMap wfd = item.getWorkflow().getWorkflowData().getMetaDataMap();
wfd.put("mykey", "My Step Value");
Set<String> keyset = wfd.keySet();
Iterator<String> i = keyset.iterator();
while (i.hasNext()){
Object key = i.next();
log.info("The workflow medata includes key {} and value {}",key.toString(),wfd.get(key).toString());
}
}
Script ECMA ecma-script
La variable graniteWorkItem
est la représentation de script ECMA de l’objet Java WorkItem
actif. Par conséquent, vous pouvez utiliser la variable graniteWorkItem
pour obtenir les métadonnées de workflow. Le script ECMA suivant peut être utilisé pour implémenter un composant Étape du processus afin d’ajouter un élément à l’objet de workflow MetaDataMap
, puis consigner chaque élément. Ces éléments sont ensuite disponibles pour les étapes suivantes du workflow.
metaData
immédiatement disponible pour le script de l’étape est la métadonnée de l’étape. Les métadonnées d’étape sont différentes des métadonnées de workflow.var currentDateInMillis = new Date().getTime();
graniteWorkItem.getWorkflowData().getMetaDataMap().put("hardcodedKey","theKey");
graniteWorkItem.getWorkflowData().getMetaDataMap().put("currentDateInMillisKey",currentDateInMillis);
var iterator = graniteWorkItem.getWorkflowData().getMetaDataMap().keySet().iterator();
while (iterator.hasNext()){
var key = iterator.next();
log.info("Workflow metadata key, value = " + key.toString() + ", " + graniteWorkItem.getWorkflowData().getMetaDataMap().get(key));
}
Accès aux valeurs des propriétés de boîte de dialogue au moment de l’exécution accessing-dialog-property-values-at-runtime
L’objet MetaDataMap
des instances de workflow est utile pour stocker et récupérer des données tout au long de la durée de vie du workflow. Dans le cas d’implémentations de composants d’étape de workflow, l’objet MetaDataMap
est particulièrement utile pour récupérer les valeurs de propriété de composant au moment de l’exécution.
Le workflow MetaDataMap
est disponible pour les implémentations de processus de script Java et ECMA :
-
Dans les implémentations Java de l’interface WorkflowProcess, le paramètre
args
est l’objetMetaDataMap
du workflow. -
Dans les implémentations de script ECMA, la valeur est disponible en utilisant les variables
args
etmetadata
.
Exemple : récupération des arguments du composant de l’étape de processus example-retrieving-the-arguments-of-the-process-step-component
La boîte de dialogue de modification du composant Étape du processus inclut la propriété Arguments. La valeur de la propriété Arguments est stockée dans les métadonnées du workflow et est associée à la clé PROCESS_ARGS
.
Dans le diagramme suivant, la valeur de la propriété Arguments est argument1, argument2
:
Java java-1
Le code Java suivant est la méthode execute
pour une implémentation WorkflowProcess
. La méthode enregistre la valeur dans l’objet args
MetaDataMap
associé à la clé PROCESS_ARGS
.
public void execute(WorkItem item, WorkflowSession session, MetaDataMap args) throws WorkflowException {
if (args.containsKey("PROCESS_ARGS")){
log.info("workflow metadata for key PROCESS_ARGS and value {}",args.get("PROCESS_ARGS","string").toString());
}
}
Lorsqu’une étape de processus qui utilise cette implémentation Java s’exécute, le journal contient l’entrée suivante :
16.02.2018 12:07:39.566 *INFO* [JobHandler: /var/workflow/instances/server0/2018-02-16/model_855140139900189:/content/we-retail/de] com.adobe.example.workflow.impl.process.LogArguments workflow metadata for key PROCESS_ARGS and value argument1, argument2
Script ECMA ecma-script-1
Le script ECMA suivant est utilisé comme processus pour Étape du processus. Il consigne le nombre d’arguments et les valeurs d’argument :
var iterator = graniteWorkItem.getWorkflowData().getMetaDataMap().keySet().iterator();
while (iterator.hasNext()){
var key = iterator.next();
log.info("Workflow metadata key, value = " + key.toString() + ", " + graniteWorkItem.getWorkflowData().getMetaDataMap().get(key));
}
log.info("hardcodedKey "+ graniteWorkItem.getWorkflowData().getMetaDataMap().get("hardcodedKey"));
log.info("currentDateInMillisKey "+ graniteWorkItem.getWorkflowData().getMetaDataMap().get("currentDateInMillisKey"));
Scripts et arguments de processus scripts-and-process-arguments
Dans un script pour un composant Étape du processus, les arguments sont disponibles via l’objet args
.
Lors de la création d’un composant d’étape personnalisée, l’objet metaData
est disponible dans un script. Cet objet est limité à un seul argument de chaîne.
Développement de mises en œuvre d’étapes de processus developing-process-step-implementations
Lorsque les étapes de processus sont lancées au cours du traitement d’un processus, les étapes envoient une requête à un service OSGi ou exécutent un script ECMA. Développez le service ou le script ECMA qui effectue les actions requises par votre workflow.
Mise en œuvre d’une étape de processus avec une classe Java implementing-a-process-step-with-a-java-class
Pour définir une étape de processus en tant que composant de service OSGI (bundle Java) :
-
Créez le bundle et déployez-le dans le conteneur OSGI. Reportez-vous à la documentation sur la création d’un bundle avec CRXDE Lite ou Eclipse.
note note NOTE Le composant OSGI doit implémenter l’interface WorkflowProcess
avec sa méthodeexecute()
. Voir l’exemple de code ci-dessous.note note NOTE Le nom du package doit être ajouté à la section <*Private-Package*>
de la configurationmaven-bundle-plugin
. -
Ajoutez la propriété SCR
process.label
et définissez sa valeur selon vos besoins. Il s’agit du nom sous lequel votre étape de processus est listée lorsque vous utilisez le composant générique Étape du processus. Voir l’exemple ci-dessous. -
Dans l’éditeur de Modèles, ajoutez l’étape de traitement au workflow à l’aide du composant générique Étape du processus.
-
Dans la boîte de dialogue de modification (de l’étape du processus), accédez à l’onglet Processus et sélectionnez votre implémentation de processus.
-
Si vous utilisez des arguments dans votre code, définissez les Arguments du processus. Par exemple : false.
-
Enregistrez les modifications, tant pour l’étape que pour le modèle de workflow (coin supérieur gauche de l’éditeur de modèles).
Les méthodes java, respectivement les classes qui implémentent la méthode Java exécutable, sont enregistrées en tant que services OSGI, ce qui vous permet d’ajouter des méthodes à tout moment pendant l’exécution.
Le composant OSGI suivant ajoute la propriété approved
au nœud de contenu de page lorsque la charge est une page :
package com.adobe.example.workflow.impl.process;
import com.adobe.granite.workflow.WorkflowException;
import com.adobe.granite.workflow.WorkflowSession;
import com.adobe.granite.workflow.exec.WorkItem;
import com.adobe.granite.workflow.exec.WorkflowData;
import com.adobe.granite.workflow.exec.WorkflowProcess;
import com.adobe.granite.workflow.metadata.MetaDataMap;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Service;
import org.osgi.framework.Constants;
import javax.jcr.Node;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
/**
* Sample workflow process that sets an <code>approve</code> property to the payload based on the process argument value.
*/
@Component
@Service
public class MyProcess implements WorkflowProcess {
@Property(value = "An example workflow process implementation.")
static final String DESCRIPTION = Constants.SERVICE_DESCRIPTION;
@Property(value = "Adobe")
static final String VENDOR = Constants.SERVICE_VENDOR;
@Property(value = "My Sample Workflow Process")
static final String LABEL="process.label";
private static final String TYPE_JCR_PATH = "JCR_PATH";
public void execute(WorkItem item, WorkflowSession session, MetaDataMap args) throws WorkflowException {
WorkflowData workflowData = item.getWorkflowData();
if (workflowData.getPayloadType().equals(TYPE_JCR_PATH)) {
String path = workflowData.getPayload().toString() + "/jcr:content";
try {
Session jcrSession = session.adaptTo(Session.class);
Node node = (Node) jcrSession.getItem(path);
if (node != null) {
node.setProperty("approved", readArgument(args));
jcrSession.save();
}
} catch (RepositoryException e) {
throw new WorkflowException(e.getMessage(), e);
}
}
}
private boolean readArgument(MetaDataMap args) {
String argument = args.get("PROCESS_ARGS", "false");
return argument.equalsIgnoreCase("true");
}
}
Utilisation d’ECMAScript using-ecmascript
Les scripts ECMA permettent aux développeurs de scripts d’implémenter des étapes de processus. Les scripts sont situés dans le référentiel JCR et exécutés à partir de cet emplacement.
Le tableau suivant répertorie les variables immédiatement disponibles pour traiter les scripts, ce qui permet d’accéder aux objets de l’API Java du workflow.
com.adobe.granite.workflow.exec.WorkItem
graniteWorkItem
com.adobe.granite.workflow.WorkflowSession
graniteWorkflowSession
String[]
(contient des arguments de processus)args
com.adobe.granite.workflow.metadata.MetaDataMap
metaData
org.apache.sling.scripting.core.impl.InternalScriptHelper
sling
L’exemple de script suivant montre comment accéder au nœud JCR qui représente le payload du workflow. La variable graniteWorkflowSession
est adaptée à une variable de session JCR, utilisée pour obtenir le nœud à partir du chemin d’accès du payload.
var workflowData = graniteWorkItem.getWorkflowData();
if (workflowData.getPayloadType() == "JCR_PATH") {
var path = workflowData.getPayload().toString();
var jcrsession = graniteWorkflowSession.adaptTo(Packages.javax.jcr.Session);
var node = jcrsession.getNode(path);
if (node.hasProperty("approved")){
node.setProperty("approved", args[0] == "true" ? true : false);
node.save();
}
}
Le script suivant vérifie si le payload est un fichier image (.png
), crée une image en noir et blanc et l’enregistre en tant que nœud frère.
var workflowData = graniteWorkItem.getWorkflowData();
if (workflowData.getPayloadType() == "JCR_PATH") {
var path = workflowData.getPayload().toString();
var jcrsession = graniteWorkflowSession.adaptTo(Packages.javax.jcr.Session);
var node = jcrsession.getRootNode().getNode(path.substring(1));
if (node.isNodeType("nt:file") && node.getProperty("jcr:content/jcr:mimeType").getString().indexOf("image/") == 0) {
var is = node.getProperty("jcr:content/jcr:data").getStream();
var layer = new Packages.com.day.image.Layer(is);
layer.grayscale();
var parent = node.getParent();
var gn = parent.addNode("grey" + node.getName(), "nt:file");
var content = gn.addNode("jcr:content", "nt:resource");
content.setProperty("jcr:mimeType","image/png");
var cal = Packages.java.util.Calendar.getInstance();
content.setProperty("jcr:lastModified",cal);
var f = Packages.java.io.File.createTempFile("test",".png");
var tout = new Packages.java.io.FileOutputStream(f);
layer.write("image/png", 1.0, tout);
var fis = new Packages.java.io.FileInputStream(f);
content.setProperty("jcr:data", fis);
parent.save();
tout.close();
fis.close();
is.close();
f.deleteOnExit();
}
}
Pour utiliser le script, procédez comme suit :
-
Créez le script (par exemple avec CRXDE Lite) et enregistrez-le dans le référentiel sous
//apps/workflow/scripts/
. -
Pour spécifier un titre qui identifie le script dans la boîte de dialogue de modification d’Étape du processus, ajoutez les propriétés suivantes au nœud
jcr:content
de votre script :table 0-row-3 1-row-3 2-row-3 Nom Type Valeur jcr:mixinTypes
Name[]
mix:title
jcr:title
String
Nom à afficher dans la boîte de dialogue de modification. -
Modifiez l’instance Étape de processus et spécifiez le script à utiliser.
Développement de programmes de sélection des participants developing-participant-choosers
Vous pouvez développer des programmes de sélection des participants pour les composants Étape du participant dynamique.
Lorsqu’un composant Étape de participant dynamique est démarré pendant un workflow, l’étape doit déterminer le participant auquel l’élément de travail généré peut être attribué. Pour ce faire, l’étape :
- envoie une requête à un service OSGi ou
- exécute un script ECMA pour sélectionner le participant ou la participante
Vous pouvez développer un service ou un script ECMA qui sélectionne le participant ou la participante en fonction des exigences de votre workflow.
Développement d’un programme de sélection des participants et participantes à l’aide d’une classe Java developing-a-participant-chooser-using-a-java-class
Pour définir une étape de participant en tant que composant de service OSGI (classe Java) :
-
Le composant OSGI doit implémenter l’interface
ParticipantStepChooser
avec sa méthodegetParticipant()
. Voir l’exemple de code ci-dessous.Créez le bundle et déployez-le dans le conteneur OSGI.
-
Ajoutez la propriété SCR
chooser.label
et définissez sa valeur selon vos besoins. Il s’agit du nom sous lequel votre programme de sélection de participants est listé si vous utilisez le composant Étape de participant dynamique. Voir l’exemple :code language-java package com.adobe.example.workflow.impl.process; import com.adobe.granite.workflow.WorkflowException; import com.adobe.granite.workflow.WorkflowSession; import com.adobe.granite.workflow.exec.ParticipantStepChooser; import com.adobe.granite.workflow.exec.WorkItem; import com.adobe.granite.workflow.exec.WorkflowData; import com.adobe.granite.workflow.metadata.MetaDataMap; import org.apache.felix.scr.annotations.Component; import org.apache.felix.scr.annotations.Property; import org.apache.felix.scr.annotations.Service; import org.osgi.framework.Constants; /** * Sample dynamic participant step that determines the participant based on a path given as argument. */ @Component @Service public class MyDynamicParticipant implements ParticipantStepChooser { @Property(value = "An example implementation of a dynamic participant chooser.") static final String DESCRIPTION = Constants.SERVICE_DESCRIPTION; @Property(value = "Adobe") static final String VENDOR = Constants.SERVICE_VENDOR; @Property(value = "Dynamic Participant Chooser Process") static final String LABEL=ParticipantStepChooser.SERVICE_PROPERTY_LABEL; private static final String TYPE_JCR_PATH = "JCR_PATH"; public String getParticipant(WorkItem workItem, WorkflowSession workflowSession, MetaDataMap args) throws WorkflowException { WorkflowData workflowData = workItem.getWorkflowData(); if (workflowData.getPayloadType().equals(TYPE_JCR_PATH)) { String path = workflowData.getPayload().toString(); String pathFromArgument = args.get("PROCESS_ARGS", String.class); if (pathFromArgument != null && path.startsWith(pathFromArgument)) { return "admin"; } } return "administrators"; } }
-
Dans l’éditeur de Modèles, ajoutez l’étape de participant dynamique au workflow à l’aide du composant Étape de participant dynamique générique.
-
Dans la boîte de dialogue de modification, sélectionnez l’onglet Programme de sélection des participants et choisissez votre implémentation de programme de sélection.
-
Si vous utilisez des arguments dans votre code, définissez les Arguments du processus. Pour cet exemple :
/content/we-retail/de
. -
Enregistrez les modifications, tant pour l’étape que pour le modèle de workflow.
Développement d’un sélecteur de participant et participante à l’aide d’un script ECMA developing-a-participant-chooser-using-an-ecma-script
Vous pouvez créer un script ECMA qui sélectionne l’utilisateur auquel est affecté l’élément de travail généré par Étape du participant. Le script doit inclure une fonction nommée getParticipant
qui ne nécessite aucun argument et renvoie une String
contenant l’ID d’un utilisateur ou d’un groupe.
Les scripts sont situés dans le référentiel JCR et exécutés à partir de cet emplacement.
Le tableau suivant répertorie les variables qui permettent un accès immédiat aux objets Java de workflow dans vos scripts.
com.adobe.granite.workflow.exec.WorkItem
graniteWorkItem
com.adobe.granite.workflow.WorkflowSession
graniteWorkflowSession
String[]
(contient des arguments de processus)args
com.adobe.granite.workflow.metadata.MetaDataMap
metaData
org.apache.sling.scripting.core.impl.InternalScriptHelper
sling
function getParticipant() {
var workflowData = graniteWorkItem.getWorkflowData();
if (workflowData.getPayloadType() == "JCR_PATH") {
var path = workflowData.getPayload().toString();
if (path.indexOf("/content/we-retail/de") == 0) {
return "admin";
} else {
return "administrators";
}
}
}
-
Créez le script (par exemple avec CRXDE Lite) et enregistrez-le dans le référentiel sous
//apps/workflow/scripts
. -
Pour spécifier un titre qui identifie le script dans la boîte de dialogue de modification d’Étape du processus, ajoutez les propriétés suivantes au nœud
jcr:content
de votre script :table 0-row-3 1-row-3 2-row-3 Nom Type Valeur jcr:mixinTypes
Name[]
mix:title
jcr:title
String
Nom à afficher dans la boîte de dialogue de modification. -
Modifiez l’instance Étape de participant dynamique et spécifiez le script à utiliser.
Gestion des packages de workflow handling-workflow-packages
Les packages de workflow peuvent être transmis à un workflow pour traitement. Les packages de workflow contiennent des références à des éléments tels que des pages et des ressources.
Vous pouvez développer des étapes de workflow qui obtiennent et traitent les ressources de package. Les membres suivants du package com.day.cq.workflow.collection
donnent accès aux packages de workflow :
ResourceCollection
: classe de package de workflow.ResourceCollectionUtil
: permet de récupérer des objets ResourceCollection.ResourceCollectionManager
: crée et récupère des collections. Une implémentation est déployée en tant que service OSGi.
L’exemple de classe Java suivant montre comment obtenir des ressources de package :
package com.adobe.example;
import java.util.ArrayList;
import java.util.List;
import com.day.cq.workflow.WorkflowException;
import com.day.cq.workflow.WorkflowSession;
import com.day.cq.workflow.collection.ResourceCollection;
import com.day.cq.workflow.collection.ResourceCollectionManager;
import com.day.cq.workflow.collection.ResourceCollectionUtil;
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;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Service;
import org.apache.felix.scr.annotations.Reference;
import org.osgi.framework.Constants;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.jcr.Node;
import javax.jcr.PathNotFoundException;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
@Component
@Service
public class LaunchBulkActivate implements WorkflowProcess {
private static final Logger log = LoggerFactory.getLogger(LaunchBulkActivate.class);
@Property(value="Bulk Activate for Launches")
static final String PROCESS_NAME ="process.label";
@Property(value="A sample workflow process step to support Launches bulk activation of pages")
static final String SERVICE_DESCRIPTION = Constants.SERVICE_DESCRIPTION;
@Reference
private ResourceCollectionManager rcManager;
public void execute(WorkItem workItem, WorkflowSession workflowSession) throws Exception {
Session session = workflowSession.getSession();
WorkflowData data = workItem.getWorkflowData();
String path = null;
String type = data.getPayloadType();
if (type.equals(TYPE_JCR_PATH) && data.getPayload() != null) {
String payloadData = (String) data.getPayload();
if (session.itemExists(payloadData)) {
path = payloadData;
}
} else if (data.getPayload() != null && type.equals(TYPE_JCR_UUID)) {
Node node = session.getNodeByUUID((String) data.getPayload());
path = node.getPath();
}
// CUSTOMIZED CODE IF REQUIRED....
if (path != null) {
// check for resource collection
ResourceCollection rcCollection = ResourceCollectionUtil.getResourceCollection((Node)session.getItem(path), rcManager);
// get list of paths to replicate (no resource collection: size == 1
// otherwise size >= 1
List<String> paths = getPaths(path, rcCollection);
for (String aPath: paths) {
// CUSTOMIZED CODE....
}
} else {
log.warn("Cannot process because path is null for this " + "workitem: " + workItem.toString());
}
}
/**
* helper
*/
private List<String> getPaths(String path, ResourceCollection rcCollection) {
List<String> paths = new ArrayList<String>();
if (rcCollection == null) {
paths.add(path);
} else {
log.debug("ResourceCollection detected " + rcCollection.getPath());
// this is a resource collection. the collection itself is not
// replicated. only its members
try {
List<Node> members = rcCollection.list(new String[]{"cq:Page", "dam:Asset"});
for (Node member: members) {
String mPath = member.getPath();
paths.add(mPath);
}
} catch(RepositoryException re) {
log.error("Cannot build path list out of the resource collection " + rcCollection.getPath());
}
}
return paths;
}
}
Exemple : création d’une étape personnalisée example-creating-a-custom-step
Pour commencer facilement à créer votre propre étape personnalisée, copiez une étape existante depuis :
/libs/cq/workflow/components/model
Création de l’étape de base creating-the-basic-step
-
Recréez le chemin sous /apps, par exemple :
/apps/cq/workflow/components/model
Les nouveaux dossiers sont de type
nt:folder
:code language-xml - apps - cq - workflow (nt:folder) - components (nt:folder) - model (nt:folder)
note note NOTE Cette étape ne s’applique pas à l’éditeur de modèle d’IU classique. -
Placez ensuite l’étape copiée dans votre dossier /apps ; par exemple :
/apps/cq/workflow/components/model/myCustomStep
Voici le résultat de notre exemple d’étape personnalisée :
note caution CAUTION Puisque dans l’IU standard, seul le titre, et non les détails, est affiché sur la carte, details.jsp
n’est pas nécessaire (comme c’était le cas avec l’éditeur de l’IU classique). -
Appliquez les propriétés suivantes au nœud :
/apps/cq/workflow/components/model/myCustomStep
Propriétés d’intérêt :
-
sling:resourceSuperType
Doit hériter d’une étape existante.
Dans cet exemple, nous héritons de l’étape de base à partir de
cq/workflow/components/model/step
, mais vous pouvez utiliser d’autres super types commeparticipant
,process
, etc. -
jcr:title
Le titre est-il affiché lorsque le composant est répertorié dans le navigateur d’étapes (panneau de gauche de l’éditeur de modèle de workflow).
-
cq:icon
Permet de spécifier une icône Coral pour l’étape.
-
componentGroup
Doit être l’un des éléments suivants :
- Workflow de collaboration
- Workflow de gestion des ressources numériques
- Workflows de formulaires
- Projets
- Workflow de gestion de contenu web
- Workflow
-
-
Vous pouvez désormais ouvrir un modèle de workflow pour le modifier. Dans le navigateur d’étapes, vous pouvez filtrer pour voir Mon étape personnalisée :
Si vous faites glisser Mon étape personnalisée sur le modèle, la carte s’affiche :
Si la propriété
cq:icon
n’a pas été définie pour l’étape, une icône par défaut est rendue d’après les deux premières lettres du titre. Par exemple :
Définition de la boîte de dialogue Configuration de l’étape defining-the-step-configure-dialog
Après la création de l’étape de base, définissez la la boîte de dialogue Configuration de l’étape comme suit :
-
Configurez les propriétés sur le nœud
cq:editConfig
comme suit :Propriétés d’intérêt :
-
cq:inherit
Si sa valeur est
true
, votre composant d’étape hérite des propriétés de l’étape spécifiée danssling:resourceSuperType
. -
cq:disableTargeting
Définissez-la selon vos besoins.
-
-
Configurez les propriétés sur le nœud
cq:formsParameter
comme suit :Propriétés d’intérêt :
-
jcr:title
Définit le titre par défaut sur la carte étape dans la carte modèle et dans le champ Titre de la boîte de dialogue de configuration Mes propriétés d’étape personnalisées.
-
Vous pouvez également définir vos propres propriétés personnalisées.
-
-
Configurez les propriétés sur le nœud
cq:listeners
.Le nœud
cq:listener
et ses propriétés vous permettent de définir des gestionnaires d’événements réagissant aux événements dans l’éditeur de modèles de l’IU tactile, de glisser une étape sur une page de modèle ou de modifier les propriétés d’une étape.Propriétés d’intérêt :
afterMove: REFRESH_PAGE
afterdelete: CQ.workflow.flow.Step.afterDelete
afteredit: CQ.workflow.flow.Step.afterEdit
afterinsert: CQ.workflow.flow.Step.afterInsert
Cette configuration est essentielle au bon fonctionnement de l’éditeur. Dans la plupart des cas, cette configuration ne doit pas être modifiée.
Toutefois, la définition de
cq:inherit
sur true (dans le nœudcq:editConfig
, tel que ci-dessus) permet d’hériter de cette configuration, sans avoir à l’inclure explicitement dans la définition de l’étape. Si aucun héritage n’est en place, vous devez ajouter ce nœud avec les propriétés et valeurs suivantes.Dans cet exemple, l’héritage a été activé pour pouvoir supprimer le nœud
cq:listeners
et permettre à l’étape de fonctionner correctement. -
Vous pouvez désormais ajouter une instance de votre étape à un modèle de workflow. Lorsque vous configurez l’étape vous voyez s’afficher la boîte de dialogue :
Exemple de balisage utilisé dans ce cas de figure sample-markup-used-in-this-example
Le balisage d’une étape personnalisée doit être représenté dans le fichier .content.xml
du composant nœud racine. Exemple de .content.xml
utilisé pour ce cas de figure :
/apps/cq/workflow/components/model/myCustomStep/.content.xml
<?xml version="1.0" encoding="UTF-8"?>
<jcr:root xmlns:sling="https://sling.apache.org/jcr/sling/1.0" xmlns:cq="https://www.day.com/jcr/cq/1.0" xmlns:jcr="https://www.jcp.org/jcr/1.0"
cq:icon="bell"
jcr:primaryType="cq:Component"
jcr:title="My Custom Step"
sling:resourceSuperType="cq/workflow/components/model/process"
allowedParents="[*/parsys]"
componentGroup="Workflow"/>
Exemple de _cq_editConfig.xml
utilisé dans ce cas de figure :
<?xml version="1.0" encoding="UTF-8"?>
<jcr:root xmlns:cq="https://www.day.com/jcr/cq/1.0" xmlns:jcr="https://www.jcp.org/jcr/1.0" xmlns:nt="https://www.jcp.org/jcr/nt/1.0"
cq:disableTargeting="{Boolean}true"
cq:inherit="{Boolean}true"
jcr:primaryType="cq:EditConfig">
<cq:formParameters
jcr:primaryType="nt:unstructured"
jcr:title="My Custom Step Card"
SAMPLE_PROPERY="sample value"/>
<cq:listeners
jcr:primaryType="cq:EditListenersConfig"
afterdelete="CQ.workflow.flow.Step.afterDelete"
afteredit="CQ.workflow.flow.Step.afterEdit"
afterinsert="CQ.workflow.flow.Step.afterInsert"
afterMove="REFRESH_PAGE"/>
</jcr:root>
Exemple de _cq_dialog/.content.xml
utilisé dans ce cas de figure :
<?xml version="1.0" encoding="UTF-8"?>
<jcr:root xmlns:sling="https://sling.apache.org/jcr/sling/1.0" xmlns:cq="https://www.day.com/jcr/cq/1.0" xmlns:jcr="https://www.jcp.org/jcr/1.0" xmlns:nt="https://www.jcp.org/jcr/nt/1.0"
jcr:primaryType="nt:unstructured"
jcr:title="My Custom - Step Properties"
sling:resourceType="cq/gui/components/authoring/dialog">
<content
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/tabs">
<items jcr:primaryType="nt:unstructured">
<common
cq:hideOnEdit="true"
jcr:primaryType="nt:unstructured"
jcr:title="Common"
sling:resourceType="granite/ui/components/coral/foundation/fixedcolumns"/>
<process
cq:hideOnEdit="true"
jcr:primaryType="nt:unstructured"
jcr:title="Process"
sling:resourceType="granite/ui/components/coral/foundation/fixedcolumns"/>
<mycommon
jcr:primaryType="nt:unstructured"
jcr:title="Common"
sling:resourceType="granite/ui/components/coral/foundation/fixedcolumns">
<items jcr:primaryType="nt:unstructured">
<columns
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/container">
<items jcr:primaryType="nt:unstructured">
<title
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/form/textfield"
fieldLabel="Title"
name="./jcr:title"/>
<description
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/form/textarea"
fieldLabel="Description"
name="./jcr:description"/>
</items>
</columns>
</items>
</mycommon>
<advanced
jcr:primaryType="nt:unstructured"
jcr:title="Advanced"
sling:resourceType="granite/ui/components/coral/foundation/fixedcolumns">
<items jcr:primaryType="nt:unstructured">
<columns
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/container">
<items jcr:primaryType="nt:unstructured">
<email
jcr:primaryType="nt:unstructured"
sling:resourceType="granite/ui/components/coral/foundation/form/checkbox"
fieldDescription="Notify user via email."
fieldLabel="Email"
name="./metaData/PROCESS_AUTO_ADVANCE"
text="Notify user via email."
value="true"/>
</items>
</columns>
</items>
</advanced>
</items>
</content>
</jcr:root>
sling:resourceSuperType : cq/workflow/components/model/process
-
Dans les cas où une boîte de dialogue mise à niveau est vide, vous pouvez consulter des boîtes de dialogue
/libs
avec des fonctionnalités similaires à titre d’exemple. Par exemple : -
/libs/cq/workflow/components/model
-
/libs/cq/workflow/components/workflow
-
/libs/dam/components
-
/libs/wcm/workflow/components/autoassign
-
/libs/cq/projects
Vous n’avez pas à modifier quoi que ce soit dans
/libs
, utilisez simplement les éléments comme exemples. Si vous souhaitez tirer parti des étapes existantes, copiez-les dans/apps
et modifiez-les.