Camunda如何发送邮件及委托代码讲解

💖专栏简介

✔️本专栏将从Camunda(卡蒙达) 7中的关键概念到实现中国式工作流相关功能。

✔️文章中只包含演示核心代码及测试数据,完整代码可查看作者的开源项目snail-camunda

✔️请给snail-camunda 点颗星吧😘

💖什么是委托代码

委托代码是在流程执行过程中发生某些事件时执行外部Java代码、脚本或求值表达式。在Camunda中分为如下4种:

  • Java Delegate 【Java委托】
  • Delegate Variable Mapping 【委托变量映射】
  • Execution Listener 【执行监听器】
  • Task Listener 【任务监听器】

💖Java Delegate

要实现一个可以在流程执行期间调用的类,该类需要实现org.camunda.bpm.engine.delegate.JavaDelegate 接口,并在execute方法中提供所需的逻辑。当流程执行到达这个特定步骤时,它将执行在该方法中定义的逻辑,并以默认的BPMN 2.0方式离开活动。

接下来通过Send Task发送邮件来演示:

<?xml version="1.0" encoding="UTF-8"?>
<bpmn:definitions xmlns:bpmn="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:camunda="http://camunda.org/schema/1.0/bpmn" xmlns:di="http://www.omg.org/spec/DD/20100524/DI" xmlns:modeler="http://camunda.org/schema/modeler/1.0" id="Definitions_19cqxxj" targetNamespace="http://bpmn.io/schema/bpmn" exporter="Camunda Modeler" exporterVersion="5.19.0" modeler:executionPlatform="Camunda Platform" modeler:executionPlatformVersion="7.15.0">
  <bpmn:process id="Process_0cu2yds" isExecutable="true">
    <bpmn:startEvent id="StartEvent_1">
      <bpmn:outgoing>Flow_0yf2ndo</bpmn:outgoing>
    </bpmn:startEvent>
    <bpmn:sequenceFlow id="Flow_0yf2ndo" sourceRef="StartEvent_1" targetRef="Activity_1jxjk7q" />
    <bpmn:userTask id="Activity_1jxjk7q" name="发起人" camunda:assignee="${initiator}">
      <bpmn:incoming>Flow_0yf2ndo</bpmn:incoming>
      <bpmn:outgoing>Flow_0ti4nj0</bpmn:outgoing>
    </bpmn:userTask>
    <bpmn:sequenceFlow id="Flow_0ti4nj0" sourceRef="Activity_1jxjk7q" targetRef="Activity_0g9hruk" />
    <bpmn:sendTask id="Activity_0g9hruk" name="戴宗报信" camunda:class="com.lonewalker.snail.delegate.SendEmailDelegate">
      <bpmn:incoming>Flow_0ti4nj0</bpmn:incoming>
      <bpmn:outgoing>Flow_0mub7f0</bpmn:outgoing>
    </bpmn:sendTask>
    <bpmn:sequenceFlow id="Flow_0mub7f0" sourceRef="Activity_0g9hruk" targetRef="Activity_111hike" />
    <bpmn:userTask id="Activity_111hike" name="宋江" camunda:assignee="${head}">
      <bpmn:incoming>Flow_0mub7f0</bpmn:incoming>
      <bpmn:outgoing>Flow_1kv7i02</bpmn:outgoing>
    </bpmn:userTask>
    <bpmn:endEvent id="Event_037pynp">
      <bpmn:incoming>Flow_1kv7i02</bpmn:incoming>
    </bpmn:endEvent>
    <bpmn:sequenceFlow id="Flow_1kv7i02" sourceRef="Activity_111hike" targetRef="Event_037pynp" />
  </bpmn:process>
  <bpmndi:BPMNDiagram id="BPMNDiagram_1">
    <bpmndi:BPMNPlane id="BPMNPlane_1" bpmnElement="Process_0cu2yds">
      <bpmndi:BPMNShape id="_BPMNShape_StartEvent_2" bpmnElement="StartEvent_1">
        <dc:Bounds x="179" y="99" width="36" height="36" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Activity_0mqjfh8_di" bpmnElement="Activity_1jxjk7q">
        <dc:Bounds x="270" y="77" width="100" height="80" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Activity_1tg1kpv_di" bpmnElement="Activity_0g9hruk">
        <dc:Bounds x="430" y="77" width="100" height="80" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Activity_1mr4xv3_di" bpmnElement="Activity_111hike">
        <dc:Bounds x="590" y="77" width="100" height="80" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Event_037pynp_di" bpmnElement="Event_037pynp">
        <dc:Bounds x="752" y="99" width="36" height="36" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNEdge id="Flow_0yf2ndo_di" bpmnElement="Flow_0yf2ndo">
        <di:waypoint x="215" y="117" />
        <di:waypoint x="270" y="117" />
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge id="Flow_0ti4nj0_di" bpmnElement="Flow_0ti4nj0">
        <di:waypoint x="370" y="117" />
        <di:waypoint x="430" y="117" />
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge id="Flow_0mub7f0_di" bpmnElement="Flow_0mub7f0">
        <di:waypoint x="530" y="117" />
        <di:waypoint x="590" y="117" />
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge id="Flow_1kv7i02_di" bpmnElement="Flow_1kv7i02">
        <di:waypoint x="690" y="117" />
        <di:waypoint x="752" y="117" />
      </bpmndi:BPMNEdge>
    </bpmndi:BPMNPlane>
  </bpmndi:BPMNDiagram>
</bpmn:definitions>

还是一样的将类路径替换为自己的。

此外,通过field injection【字段注入】设置目标邮箱,字段注入的Name建议使用驼峰命名,因为我们在代码中需要使用。

关于邮箱的设置可查看此篇【Springboot发送邮件】

在配置文件中新增邮件相关配置

###邮件相关配置
#邮箱服务器地址
spring.mail.host=smtp.163.com
#邮箱用户名
spring.mail.username=xxx@163.com
#邮箱密码(注意:qq邮箱应该使用独立密码,去qq邮箱设置里面获取 163邮箱使用设置的授权码)
#这里把授权码改为自己的
spring.mail.password=QJBFXYUNMQYRCGSU
##编码格式
spring.mail.default-encoding=UTF-8
##端口号   465或者994
spring.mail.port=465
spring.mail.protocol=smtps
##发送邮件地址
spring.mail.from=xxx@163.com

新增发送邮件的委托代码

@RequiredArgsConstructor
@Slf4j
@Component
public class SendEmailDelegate implements JavaDelegate {

    private final JavaMailSender mailSender;

    @Value("${spring.mail.from}")
    private String addresser;

    private Expression sendingTarget;

    @Override
    public void execute(DelegateExecution execution) throws Exception {
        String initiator = (String) execution.getVariable("initiator");
        String processInstanceId = execution.getProcessInstanceId();
        String content = initiator+"--发起的审批流程抄送给您["+processInstanceId+"]";

        //获取收件人,也可以在表单设计时存入数据库
        String recipient = String.valueOf(sendingTarget.getValue(execution));
        //改成自己的邮箱账号调试
        SimpleMailMessage mailMessage = new SimpleMailMessage();
        //设置发件人
        mailMessage.setFrom(addresser);
        //设置收件人
        mailMessage.setTo(recipient);
        //设置主题
        mailMessage.setSubject("审批抄送");
        //设置内容
        mailMessage.setText(content);
        //发送邮件
        mailSender.send(mailMessage);
    }
}

测试结果:

在流程定义中引用的类(例如上述的com.lonewalker.demo.delegate.SendEmailDelegate)不会在部署期间实例化。只有当流程执行到达流程中第一次使用该类的点时才会创建该类的实例。如果找不到该类,将抛出ProcessEngineException。

每次执行委托类引用活动时,将创建该类的一个单独实例。这意味着每次执行一个活动时,都会使用类的另一个实例来调用execute(DelegateExecution)。

💖Field Injection

在上述流程定义设计时已经演示了如何使用Field Injection。实际上就是把我们需要的值注入到委托类的字段中。

如果可用,则通过委托类上的公共setter方法注入该值,遵循Java Bean命名约定。

不管在流程定义中声明的值的类型是什么,注入目标上的字段的类型应该始终是

org. camunda.bbpm .engine.delegate. expression

💖其他

委托变量映射很少用就略过了,而执行监听器和任务监听器是重点,放在下一篇详细讲解Camunda执行监听器与任务监听器-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/380495.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MATLAB实现LSTM时间序列预测

LSTM模型可以在一定程度上学习和预测非平稳的时间序列,其具有强大的记忆和非线性建模能力,可以捕捉到时间序列中的复杂模式和趋势[4]。在这种情况下,LSTM模型可能会自动学习到时间序列的非平稳性,并在预测中进行适当的调整。其作为循环神经网络(RNN)的特殊形式,继承了循…

学习Pytorch深度学习运行AlexNet代码时关于在Pycharm中解决 “t >= 0 t < n_classes” 的断言错误方法

在学习深度学习的过程中&#xff0c;遇到了一个报错&#xff1a; 这跑的代码是AlexNet的代码实现。 运行时出现报错&#xff1a; C:\cb\pytorch_1000000000000\work\aten\src\ATen\native\cuda\Loss.cu:257: block: [0,0,0], thread: [4,0,0] Assertion t > 0 && t…

抽象springBoot报错

Failed to configure a DataSource: url attribute is not specified and no embedded datasource could be configured. 中文翻译&#xff1a;无法配置DataSource&#xff1a;未指定“url”属性&#xff0c;并且无法配置嵌入数据源。 DataSource 翻译&#xff1a;数据源 得…

数据结构入门(1)数据结构介绍

目录 前言 1. 什么是数据结构&#xff1f; 2.什么是算法&#xff1f; 3.数据结构和算法的重要性 前言 本文将开始介绍计算机里的数据结构。 数据结构是指数据对象中元素之间的关系&#xff0c;以及对这些关系的操作。数据结构可以分为线性结构和非线性结构。 线性结构是…

Python相关的基础模块

Python相关的基础模块 在编写远程控制工具之前&#xff0c;先要介绍用Python编写远程控制工具时所需要的 相关模块&#xff0c;为接下来编写工具打下基础。 1.subprocess模块 subprocess模块的主要作用是执行外部的命令和程序。当我们运行Python的时 候&#xff0c;其实也是在运…

32串口数据包

目录 一.数据包格式 &#xff08;1&#xff09;HEX数据包 &#xff08;2&#xff09;文本数据包 二.代码实现 &#xff08;1&#xff09;串口收发HEX数据包 (2)串口收发文本数据包&#xff08;该程序没有写出来&#xff0c;暂时找不到错误&#xff0c;以后再看&#xff09;…

蓝桥杯每日一练(python)B组

###来源于dotcpp的蓝桥杯真题 题目 2735: 蓝桥杯2022年第十三届决赛真题-取模&#xff08;Python组&#xff09; 给定 n, m &#xff0c;问是否存在两个不同的数 x, y 使得 1 ≤ x < y ≤ m 且 n mod x n mod y 。 输入格式&#xff1a; 输入包含多组独立的询问。 第一…

浅谈应该遵守的伦敦银交易规则

做伦敦银投资的朋友应遵守伦敦银交易规则&#xff0c;伦敦银交易规则不是指那些伦敦银交易技巧&#xff0c;而是在这个市场中要遵循的一些约定&#xff0c;下面我们就来讨论一下。 风险管理。风险管理即指投资者控制自己一笔乃至整体交易的风险&#xff0c;没有风险管理意识的投…

技术精英求职必备:Java开发工程师简历制作全指南

投简历找工作嘛&#xff0c;这事儿其实就跟相亲差不多&#xff0c;得让对方一眼就看上你。 在这场职场的‘相亲’中&#xff0c;怎样才能让你的简历脱颖而出&#xff0c;成为HR眼中的理想‘对象’呢&#xff1f;来&#xff0c;我给你支几招&#xff0c;让你的简历更吸引人。 …

前端又又出新框架,这次没有打包了

最近&#xff0c;前端开发领域又迎来了一个新框架——ofa.js。它的独特之处在于&#xff0c;不依赖于现有的 nodes/npm/webpack 前端开发工作流程。与jQuery类似&#xff0c;只需引用一个脚本&#xff0c;您就能像使用React/Vue/Angular一样轻松地开发大型应用。 极易上手 如果…

八卦图与二进制

名称二进制乾111坤000震100艮001离101坎010兑110巽011 1.卦象从下往上排&#xff0c;称为初爻、二爻、上爻&#xff0c;长线为1&#xff0c;短线为0&#xff0c;可以根据卦象记忆对应的二进制&#xff0c;二进制数也从下往上排。 2.注意&#xff1a;在使用二进制时&#xff0…

【精选】java初识多态 多态调用成员的特点

&#x1f36c; 博主介绍&#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 hacker-routing &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【应急响应】 【python】 【VulnHub靶场复现】【面试分析】 &#x1f389;点赞➕评论➕收藏…

SQL拆分字段内容(含分隔符)

问题描述&#xff1a; 在做数据迁移的过程中&#xff0c;我们希望对表中的某个字段根据分隔符进行拆分&#xff0c;得到多条数据&#xff0c;原代码有点意思&#xff0c;因此记录一下。 我们假设某条数据如下&#xff1a; IDSTRS1公司名称不能小于四个字&#xff0c;行业类别…

Unity学习笔记之【IK反向动力学操作】

反向动力学Inverse Kinematics 反向动力学&#xff0c;简称IK。相较于正向动力学&#xff0c;反向动力学旨在子级对父级产生的影响。 使用IK&#xff0c;可以实现根据目标位置或方向来计算并调整角色的关节&#xff08;骨骼&#xff09;链&#xff0c;以使角色的末端&#xff…

关节点检测

https://www.bilibili.com/video/BV19g4y1777q/?p2&spm_id_frompageDriver 关节点检测全流程 YOLO:单阶段&#xff0c;快&#xff1b; MMPose&#xff1a;双阶段&#xff0c;准&#xff1b; 标注工具Labelme 用Labelme标注样本数据集

FastDFS 分布式集群搭建详解

文章目录 前言1、整体架构2、安装配置FastDFS集群2.1 配置tracker2.2 配置storage 3、启动集群4、查看集群情况5、nginx配置5.1 配置storage的四台机器的nginx5.2 配置tracker的两台机器的nginx5.3 配置统一入口 前言 阅读本文章之前请先看上一篇单机版FastDFS安装配置详解&am…

为什么在产品设计和制造过程中要采用FMEA——SunFMEA软件

在产品设计和制造过程中&#xff0c;FMEA是一种非常重要的工具&#xff0c;用于评估潜在的故障模式及其对产品性能的影响。通过分析产品设计或流程中可能出现的故障模式&#xff0c;并评估其对产品性能和客户满意度的潜在影响&#xff0c;来预测和防止产品在生产和运行过程中出…

深入解析Linux中HTTP代理的工作原理

亲爱的Linux探险家们&#xff0c;准备好一起探索HTTP代理背后的神秘面纱了吗&#xff1f;在这个数字世界里&#xff0c;HTTP代理就像是一个神秘的中间人&#xff0c;默默地在你和互联网之间穿梭&#xff0c;为你传递信息。那么&#xff0c;这个神秘的中间人到底是如何工作的呢&…

vue3 的setup和生命周期

vue3 的setup和生命周期 许多文章认为setup执行时间在beforeCreate 和created 之间&#xff0c;但是通过实际测试发现setup调用在beforecreate之前。 export default {beforeCreate() {console.log(beforeCreate running....);},created() {console.log("created runnin…

制度下降算法c语言

#include<stdio.h> #include<string.h> int location0; //遍历字符串的当前位置 char arr[20]"idid*id#"; void error(); //错误提示函数 /* 每一个非终结符都构造一个函数 */ void E(char t); void Ep(char t); void T(char t); void Tp(char t);…