|
|
|
@ -3,25 +3,35 @@ package cn.iocoder.yudao.module.bpm.service.definition;
|
|
|
|
import cn.hutool.core.collection.CollUtil;
|
|
|
|
import cn.hutool.core.collection.CollUtil;
|
|
|
|
import cn.hutool.core.util.ObjectUtil;
|
|
|
|
import cn.hutool.core.util.ObjectUtil;
|
|
|
|
import cn.hutool.core.util.StrUtil;
|
|
|
|
import cn.hutool.core.util.StrUtil;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.framework.common.enums.CommonStatusEnum;
|
|
|
|
import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils;
|
|
|
|
import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils;
|
|
|
|
import cn.iocoder.yudao.framework.common.util.object.ObjectUtils;
|
|
|
|
import cn.iocoder.yudao.framework.common.util.object.ObjectUtils;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.framework.datapermission.core.annotation.DataPermission;
|
|
|
|
import cn.iocoder.yudao.framework.flowable.core.util.FlowableUtils;
|
|
|
|
import cn.iocoder.yudao.framework.flowable.core.util.FlowableUtils;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleCreateReqVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleCreateReqVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleRespVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleRespVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleUpdateReqVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.controller.admin.definition.vo.rule.BpmTaskAssignRuleUpdateReqVO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.convert.definition.BpmTaskAssignRuleConvert;
|
|
|
|
import cn.iocoder.yudao.module.bpm.convert.definition.BpmTaskAssignRuleConvert;
|
|
|
|
import cn.iocoder.yudao.module.bpm.dal.dataobject.definition.BpmTaskAssignRuleDO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.dal.dataobject.definition.BpmTaskAssignRuleDO;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.bpm.dal.dataobject.definition.BpmUserGroupDO;
|
|
|
|
import cn.iocoder.yudao.module.bpm.dal.mysql.definition.BpmTaskAssignRuleMapper;
|
|
|
|
import cn.iocoder.yudao.module.bpm.dal.mysql.definition.BpmTaskAssignRuleMapper;
|
|
|
|
import cn.iocoder.yudao.module.bpm.enums.DictTypeConstants;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.bpm.domain.enums.definition.BpmTaskAssignRuleTypeEnum;
|
|
|
|
import cn.iocoder.yudao.module.bpm.domain.enums.definition.BpmTaskAssignRuleTypeEnum;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.bpm.enums.DictTypeConstants;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.bpm.framework.flowable.core.behavior.script.BpmTaskAssignScript;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dept.DeptApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dept.DeptApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dept.PostApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dept.PostApi;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.dept.dto.DeptRespDTO;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dict.DictDataApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.dict.DictDataApi;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.permission.PermissionApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.permission.RoleApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.permission.RoleApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.user.AdminUserApi;
|
|
|
|
import cn.iocoder.yudao.module.system.api.user.AdminUserApi;
|
|
|
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.user.dto.AdminUserRespDTO;
|
|
|
|
|
|
|
|
import com.google.common.annotations.VisibleForTesting;
|
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
import org.flowable.bpmn.model.BpmnModel;
|
|
|
|
import org.flowable.bpmn.model.BpmnModel;
|
|
|
|
import org.flowable.bpmn.model.UserTask;
|
|
|
|
import org.flowable.bpmn.model.UserTask;
|
|
|
|
|
|
|
|
import org.flowable.common.engine.api.FlowableException;
|
|
|
|
|
|
|
|
import org.flowable.task.service.impl.persistence.entity.TaskEntity;
|
|
|
|
import org.springframework.context.annotation.Lazy;
|
|
|
|
import org.springframework.context.annotation.Lazy;
|
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
import org.springframework.validation.annotation.Validated;
|
|
|
|
import org.springframework.validation.annotation.Validated;
|
|
|
|
@ -31,6 +41,9 @@ import javax.validation.Valid;
|
|
|
|
import java.util.*;
|
|
|
|
import java.util.*;
|
|
|
|
|
|
|
|
|
|
|
|
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
|
|
|
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
|
|
|
|
|
|
|
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertMap;
|
|
|
|
|
|
|
|
import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertSet;
|
|
|
|
|
|
|
|
import static cn.iocoder.yudao.framework.common.util.json.JsonUtils.toJsonString;
|
|
|
|
import static cn.iocoder.yudao.module.bpm.enums.ErrorCodeConstants.*;
|
|
|
|
import static cn.iocoder.yudao.module.bpm.enums.ErrorCodeConstants.*;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
@ -61,6 +74,17 @@ public class BpmTaskAssignRuleServiceImpl implements BpmTaskAssignRuleService {
|
|
|
|
private AdminUserApi adminUserApi;
|
|
|
|
private AdminUserApi adminUserApi;
|
|
|
|
@Resource
|
|
|
|
@Resource
|
|
|
|
private DictDataApi dictDataApi;
|
|
|
|
private DictDataApi dictDataApi;
|
|
|
|
|
|
|
|
@Resource
|
|
|
|
|
|
|
|
private PermissionApi permissionApi;
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* 任务分配脚本
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
private Map<Long, BpmTaskAssignScript> scriptMap = Collections.emptyMap();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Resource
|
|
|
|
|
|
|
|
public void setScripts(List<BpmTaskAssignScript> scripts) {
|
|
|
|
|
|
|
|
this.scriptMap = convertMap(scripts, script -> script.getEnum().getId());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
@Override
|
|
|
|
public List<BpmTaskAssignRuleDO> getTaskAssignRuleListByProcessDefinitionId(String processDefinitionId,
|
|
|
|
public List<BpmTaskAssignRuleDO> getTaskAssignRuleListByProcessDefinitionId(String processDefinitionId,
|
|
|
|
@ -198,10 +222,6 @@ public class BpmTaskAssignRuleServiceImpl implements BpmTaskAssignRuleService {
|
|
|
|
adminUserApi.validUsers(options);
|
|
|
|
adminUserApi.validUsers(options);
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER_GROUP.getType())) {
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER_GROUP.getType())) {
|
|
|
|
userGroupService.validUserGroups(options);
|
|
|
|
userGroupService.validUserGroups(options);
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER_SIGN.getType())) {
|
|
|
|
|
|
|
|
adminUserApi.validUsers(options);
|
|
|
|
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.USER_OR_SIGN.getType())) {
|
|
|
|
|
|
|
|
adminUserApi.validUsers(options);
|
|
|
|
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.SCRIPT.getType())) {
|
|
|
|
} else if (Objects.equals(type, BpmTaskAssignRuleTypeEnum.SCRIPT.getType())) {
|
|
|
|
dictDataApi.validDictDatas(DictTypeConstants.TASK_ASSIGN_SCRIPT,
|
|
|
|
dictDataApi.validDictDatas(DictTypeConstants.TASK_ASSIGN_SCRIPT,
|
|
|
|
CollectionUtils.convertSet(options, String::valueOf));
|
|
|
|
CollectionUtils.convertSet(options, String::valueOf));
|
|
|
|
@ -209,4 +229,117 @@ public class BpmTaskAssignRuleServiceImpl implements BpmTaskAssignRuleService {
|
|
|
|
throw new IllegalArgumentException(StrUtil.format("未知的规则类型({})", type));
|
|
|
|
throw new IllegalArgumentException(StrUtil.format("未知的规则类型({})", type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
|
|
|
@DataPermission(enable = false) // 忽略数据权限,不然分配会存在问题
|
|
|
|
|
|
|
|
public Set<Long> calculateTaskCandidateUsers(TaskEntity task) {
|
|
|
|
|
|
|
|
BpmTaskAssignRuleDO rule = getTaskRule(task);
|
|
|
|
|
|
|
|
return calculateTaskCandidateUsers(task, rule);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@VisibleForTesting
|
|
|
|
|
|
|
|
BpmTaskAssignRuleDO getTaskRule(TaskEntity task) {
|
|
|
|
|
|
|
|
List<BpmTaskAssignRuleDO> taskRules = getTaskAssignRuleListByProcessDefinitionId(
|
|
|
|
|
|
|
|
task.getProcessDefinitionId(), task.getTaskDefinitionKey());
|
|
|
|
|
|
|
|
if (CollUtil.isEmpty(taskRules)) {
|
|
|
|
|
|
|
|
throw new FlowableException(
|
|
|
|
|
|
|
|
StrUtil.format("流程任务({}/{}/{}) 找不到符合的任务规则", task.getId(), task.getProcessDefinitionId(),
|
|
|
|
|
|
|
|
task.getTaskDefinitionKey()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (taskRules.size() > 1) {
|
|
|
|
|
|
|
|
throw new FlowableException(
|
|
|
|
|
|
|
|
StrUtil.format("流程任务({}/{}/{}) 找到过多任务规则({})", task.getId(), task.getProcessDefinitionId(),
|
|
|
|
|
|
|
|
task.getTaskDefinitionKey(), taskRules.size()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return taskRules.get(0);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@VisibleForTesting
|
|
|
|
|
|
|
|
Set<Long> calculateTaskCandidateUsers(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
Set<Long> assigneeUserIds = null;
|
|
|
|
|
|
|
|
if (Objects.equals(BpmTaskAssignRuleTypeEnum.ROLE.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByRole(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.DEPT_MEMBER.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByDeptMember(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.DEPT_LEADER.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByDeptLeader(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.POST.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByPost(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.USER.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByUser(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.USER_GROUP.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByUserGroup(task, rule);
|
|
|
|
|
|
|
|
} else if (Objects.equals(BpmTaskAssignRuleTypeEnum.SCRIPT.getType(), rule.getType())) {
|
|
|
|
|
|
|
|
assigneeUserIds = calculateTaskCandidateUsersByScript(task, rule);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 移除被禁用的用户
|
|
|
|
|
|
|
|
removeDisableUsers(assigneeUserIds);
|
|
|
|
|
|
|
|
// 如果候选人为空,抛出异常 TODO 芋艿:没候选人的策略选择。1 - 挂起;2 - 直接结束;3 - 强制一个兜底人
|
|
|
|
|
|
|
|
if (CollUtil.isEmpty(assigneeUserIds)) {
|
|
|
|
|
|
|
|
log.error("[calculateTaskCandidateUsers][流程任务({}/{}/{}) 任务规则({}) 找不到候选人]", task.getId(),
|
|
|
|
|
|
|
|
task.getProcessDefinitionId(), task.getTaskDefinitionKey(), toJsonString(rule));
|
|
|
|
|
|
|
|
throw exception(TASK_CREATE_FAIL_NO_CANDIDATE_USER);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return assigneeUserIds;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByRole(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
return permissionApi.getUserRoleIdListByRoleIds(rule.getOptions());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByDeptMember(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
List<AdminUserRespDTO> users = adminUserApi.getUsersByDeptIds(rule.getOptions());
|
|
|
|
|
|
|
|
return convertSet(users, AdminUserRespDTO::getId);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByDeptLeader(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
List<DeptRespDTO> depts = deptApi.getDepts(rule.getOptions());
|
|
|
|
|
|
|
|
return convertSet(depts, DeptRespDTO::getLeaderUserId);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByPost(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
List<AdminUserRespDTO> users = adminUserApi.getUsersByPostIds(rule.getOptions());
|
|
|
|
|
|
|
|
return convertSet(users, AdminUserRespDTO::getId);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByUser(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
return rule.getOptions();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByUserGroup(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
List<BpmUserGroupDO> userGroups = userGroupService.getUserGroupList(rule.getOptions());
|
|
|
|
|
|
|
|
Set<Long> userIds = new HashSet<>();
|
|
|
|
|
|
|
|
userGroups.forEach(group -> userIds.addAll(group.getMemberUserIds()));
|
|
|
|
|
|
|
|
return userIds;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Set<Long> calculateTaskCandidateUsersByScript(TaskEntity task, BpmTaskAssignRuleDO rule) {
|
|
|
|
|
|
|
|
// 获得对应的脚本
|
|
|
|
|
|
|
|
List<BpmTaskAssignScript> scripts = new ArrayList<>(rule.getOptions().size());
|
|
|
|
|
|
|
|
rule.getOptions().forEach(id -> {
|
|
|
|
|
|
|
|
BpmTaskAssignScript script = scriptMap.get(id);
|
|
|
|
|
|
|
|
if (script == null) {
|
|
|
|
|
|
|
|
throw exception(TASK_ASSIGN_SCRIPT_NOT_EXISTS, id);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
scripts.add(script);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
// 逐个计算任务
|
|
|
|
|
|
|
|
Set<Long> userIds = new HashSet<>();
|
|
|
|
|
|
|
|
scripts.forEach(script -> CollUtil.addAll(userIds, script.calculateTaskCandidateUsers(task)));
|
|
|
|
|
|
|
|
return userIds;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@VisibleForTesting
|
|
|
|
|
|
|
|
void removeDisableUsers(Set<Long> assigneeUserIds) {
|
|
|
|
|
|
|
|
if (CollUtil.isEmpty(assigneeUserIds)) {
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Map<Long, AdminUserRespDTO> userMap = adminUserApi.getUserMap(assigneeUserIds);
|
|
|
|
|
|
|
|
assigneeUserIds.removeIf(id -> {
|
|
|
|
|
|
|
|
AdminUserRespDTO user = userMap.get(id);
|
|
|
|
|
|
|
|
return user == null || !CommonStatusEnum.ENABLE.getStatus().equals(user.getStatus());
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|