Flowable7.x学习笔记(十七)审批我的待办

前言

        前文完成了我的待办的查询功能,本文就在此基础上从源码解读到完成审批任务的功能,审批界面我就先不带表单,直接单纯审批通过,这里需要注意的事,审批的表单其实每个节点都可能需要不同的表单内容,后续要实现自定义动态表单。

一、TaskService.complete(taskId) 的核心逻辑

① 方法签名定位

        多种重载最终委托给 TaskServiceImpl.completeInternal。

② 加载与校验任务

        从数据库中读取任务实体并检测其状态。

③ 流程/本地变量处理

        将用户在完成时传入的变量写入运行时或本地作用域。

④ 事件派发与历史记录

        在任务完成前后触发引擎事件,并将运行时与历史数据持久化。

⑤ 流程执行推进

        调用执行实体(ExecutionEntity)的 signal() 或 BPMN 行为的 leave(),推动流程继续。

⑥ 任务删除与身份链接清理

        移除任务记录及其在身份链接表中的关联。

⑦ 异步后置逻辑

        根据异步配置生成作业(JobEntity),由引擎后续调度执行。

二、方法签名定位

        在 TaskService 接口中,complete 提供多种重载,如:

void complete(String taskId);

void complete(String taskId, Map<String,Object> variables);

void complete(String taskId, String userId, Map<String,Object> variables);

        所有重载方法最终委托给实现类 org.flowable.engine.impl.TaskServiceImpl 的 completeInternal(...) 方法,以统一执行流程。 

        本文使用的是第一种重载方法。

三、加载与校验任务

        加载任务实体,从运行时任务表(ACT_RU_TASK)中读取对应 TaskEntity 对象:

TaskEntity task = taskEntityManager.findById(taskId);

        存在性校验,若无此任务,抛出 FlowableObjectNotFoundException。

if (task == null) {
    throw new FlowableObjectNotFoundException("No task found with id '" + taskId + "'");
}

         委派状态校验,拒绝完成处于“待确认”(PENDING)状态的委派任务。

if (task.getDelegationState() == DelegationState.PENDING) {
    throw new FlowableException("Cannot complete a task that is in delegation state PENDING");
}

四、流程/本地变量处理

① 流程变量写入

        默认将变量保存在流程实例作用域,调用 VariableService 完成存储。

② 本地变量写入

        若传入参数指定 localScoped=true,则写入任务本地范围。

③ 持久化历史变量

        若开启历史记录,变量的写入会在 ACT_HI_VARINST 表中留存审计轨迹。

五、事件派发与历史记录

① 事件分发

        在任务完成前后,通过 EventDispatcher 发布 TASK_COMPLETED 等引擎事件,供监听器或审计插件使用。

② 历史任务记录

        若引擎历史级别(historyLevel) ≥ AUDIT,会在 ACT_HI_TASKINST 表中记录任务结束信息(结束时间、原因等)。

③ 事务一致性

        事件与历史写入与运行时数据同一事务提交,避免不一致风险。

六、流程执行推进

① 获取执行实体

        若任务绑定在流程节点上,execution 不为 null。

ExecutionEntity execution = task.getExecution();

② 调用 BPMN 行为

        对用户任务,执行 UserTaskActivityBehavior.leave(execution),其内部会调用一下方法,从而沿着流程定义的连线推进到下一个节点。

execution.signal(null, null);

七、任务删除与身份链接清理

① 删除任务记录

        将任务从 ACT_RU_TASK 表中移除。

taskEntityManager.delete(task);

② 移除身份链接

        同时清理 ACT_RU_IDENTITYLINK 表中与该任务相关的用户/组关联,防止孤立数据。

八、异步后置逻辑

① 异步任务创建

        若任务或流程节点配置了异步 (async="true"),完成时会生成新的 JobEntity。

② 调度执行

        JobExecutor 后续会在独立线程中触发这些异步作业,解耦长耗时操作,提升吞吐。

③ 异步历史

        若开启异步历史,同步写入历史的操作也会改为异步,进一步优化性能。

九、完成后端接口

① 定义请求参数

        在前文中,我们已经在查询我的待办任务数据中,包含了任务ID,所以直接传递当前行数据中的任务ID给到后端执行审批操作即可。

package com.ceair.entity.request;

import lombok.Data;

import java.io.Serial;
import java.io.Serializable;

/**
 * @author wangbaohai
 * @ClassName ApprovalMyTaskReq
 * @description: 审批我的任务请求参数
 * @date 2025年05月02日
 * @version: 1.0.0
 */
@Data
public class ApprovalMyTaskReq implements Serializable {

    @Serial
    private static final long serialVersionUID = 1L;

    // 任务编号
    private String taskId;

}

② 创建后端接口

        接口部分主要就是使用 TaskService.complete(taskId) 这个重载方法。

/**
 * 审批我的待办任务。
 * <p>
 * 权限: /api/v1/myTask/approvalMyTask
 * 参数: approvalMyTaskReq - 包含任务ID的审批请求对象
 * 返回: Result<Boolean> 表示审批是否成功
 * <p>
 * 异常处理:
 * - 任务ID为空 → 参数错误
 * - Flowable异常 → 返回审批失败信息
 * - 其他异常 → 系统异常提示
 */
@PreAuthorize("hasAnyAuthority('/api/v1/myTask/approvalMyTask')")
@Parameter(name = "approvalMyTaskReq", description = "审批我的待办任务请求对象", required = true)
@Operation(summary = "审批我的待办任务")
@PostMapping("/approvalMyTask")
public Result<Boolean> approvalMyTask(@RequestBody ApprovalMyTaskReq approvalMyTaskReq) {
    // 参数校验
    if (approvalMyTaskReq == null || approvalMyTaskReq.getTaskId() == null) {
        log.warn("审批失败:非法的任务ID");
        return Result.error("参数不能为空或任务ID无效");
    }

    try {
        // 调用Flowable 任务服务完成任务审批
        taskService.complete(approvalMyTaskReq.getTaskId());
        return Result.success(true);
    } catch (FlowableException | IllegalArgumentException e) {
        log.error("审批我的待办任务失败,原因:{}", e.getMessage(), e);
        return Result.error("审批我的待办任务失败,原因:" + e.getMessage());
    } catch (Exception e) {
        log.error("发生未知异常,审批失败:", e);
        return Result.error("系统异常,请联系管理员");
    }
}

十、完成前端按钮功能

① 定义前端类型

// 审批任务请求参数

export interface ApprovalMyTaskReq {

  taskId: string // 任务编号,对应 Java 中的 String taskId

}

② 封装请求接口

/**

 * 审批待办任务

 */

export function approvalMyTask(data: ApprovalMyTaskReq) {

  return request.post<any>({

    url: '/pm-process/api/v1/myTask/approvalMyTask',

    data,

  })

}

③ 完善按钮界面

<el-button v-if="scope.row.status === 0 || scope.row.status === 2" v-hasButton="`btn.myTask.approvalMyTask`" type="primary" @click="onApproval(scope.row)">
  审批
</el-button>

④ 创建审批按钮方法

/**
 * 异步函数:处理任务审批
 *
 * @param row 任务对象,包含任务ID等信息
 */
async function onApproval(row: TaskVO) {
  try {
    // 获取当前任务ID并设置参数
    const param: ApprovalMyTaskReq = {
      taskId: row.taskId,
    }

    // 调用后端接口进行审批操作
    const result: any = await approvalMyTask(param)

    // 如果接口调用成功且返回的状态码为200,则显示成功提示信息
    if (result.success && result.code === 200) {
      ElMessage({
        message: '审批成功',
        type: 'success',
      })
    }
    else {
      ElMessage({
        message: `审批失败: ${result.message}`,
        type: 'error',
      })
    }
  }
  catch (error) {
    // 捕获异常并提取错误信息
    let errorMessage = '未知错误'
    if (error instanceof Error) {
      errorMessage = error.message
    }

    // 显示操作失败的错误提示信息
    ElMessage({
      message: `审批失败: ${errorMessage || '未知错误'}`,
      type: 'error',
    })
  }
}

十一、创建权限并分配

注意给至少两个用户分配角色权限

十二、测试&验证

        我们用admin用户作为第一个节点审批人,BOB用户作为第二个节点审批人,当admin审批后,我们应该切换BOB用户能看到待办任务。

发布之后我们看部署的流程的xml

然后我们启动这个最新的流程

保持当前admin账号查看我的待办任务

然后办理任务

成功之后我们切换用户到BOB

可以看到刚才的任务已经扭转到BOB帐号了

后记

        下一篇文章来梳理拾取任务的梳理以及具体的实现方法,本文的完整代码仓库地址请查看专栏第一篇文章的说明。

本文的后端分支是 process-10

本文的前端分支是 process-12

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值