using DS.Module.Core; using DS.Module.Core.Helpers; using DS.WMS.Core.Fee.Dtos; using DS.WMS.Core.Flow.Dtos; using DS.WMS.Core.Flow.Entity; using DS.WMS.Core.Flow.Interface; using DS.WMS.Core.Flow.Method; using DS.WMS.Core.Op.Dtos.TaskInteraction; using DS.WMS.Core.Op.Entity; using DS.WMS.Core.Op.Entity.TaskInteraction; using DS.WMS.Core.Op.Interface.TaskInteraction; using DS.WMS.Core.Sys.Entity; using DS.WMS.Core.TaskPlat.Dtos; using DS.WMS.Core.TaskPlat.Entity; using DS.WMS.Core.TaskPlat.Interface; using Masuit.Tools; using Masuit.Tools.Systems; using Microsoft.Extensions.DependencyInjection; using Org.BouncyCastle.Crypto; using SqlSugar; namespace DS.WMS.Core.Op.Method.TaskInteraction { /// /// 任务交互服务 /// public abstract class TaskService : ServiceBase, ITaskService { const long PERMISSION_ID = 1815294400855674880; /// /// 任务管理服务 /// protected ITaskManageService ManagerService { get; private set; } /// /// 工作流服务 /// protected Lazy FlowService { get; private set; } /// /// 初始化 /// /// public TaskService(IServiceProvider provider) : base(provider) { ManagerService = provider.GetRequiredService(); FlowService = new Lazy(provider.GetRequiredService()); } /// /// 获取给定任务的下一任务类型 /// /// 任务信息 /// internal static TaskBaseTypeEnum? GetNextType(BusinessTask current) { if (current.TaskType == TaskBaseTypeEnum.NOT_SPECIFIED || current.TaskType == TaskBaseTypeEnum.WAIT_CHECKOUT_BILL) //流程的最后一步 return null; int currentTypeVal = (int)current.TaskType; if (currentTypeVal >= 300) //300开始的枚举值为可选服务项目,不存在前后关联性 return null; return (TaskBaseTypeEnum)(currentTypeVal + 1); } /// /// 获取关联业务的所有任务 /// /// 业务ID /// 业务类型 /// public async Task>> GetTasks(long id, BusinessType businessType) { var list = await TenantDb.Queryable().Where( x => x.BusinessId == id && x.BusinessType == businessType).Select().ToListAsync(); var result = DataResult>.Success(list); result.Count = list.Count; return result; } /// /// 确保任务交互模块已授权 /// /// protected virtual async Task EnsureModuleAuthorized() { long tid = long.Parse(User.TenantId); var authStr = await Db.Queryable().Where(x => x.PermissionId == PERMISSION_ID && x.TenantId == tid && SqlFunc.Subqueryable().Where(spt => spt.PermissionId == x.PermissionId).Any()) .Select(x => x.AuthNum).FirstAsync(); if (authStr.IsNullOrEmpty()) return false; var appSecret = await Db.Queryable().Where(x => x.Id == tid).Select(x => x.AppSecret).FirstAsync(); return int.TryParse(EncrypteHelper.DecryptData(authStr, appSecret), out int authNum) && authNum > 0; } /// /// 发起任务审核 /// /// /// public async Task SubmitAuditAsync(TaskRequest request) { var task = await GetQuery(request.BusinessId, request.BusinessType, TaskBaseTypeEnum.WAIT_ORDER_AUDIT).FirstAsync(); if (task == null) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.EmptyData)); if (task.TaskStatus == TaskStatusEnum.Complete) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskCompleted)); if (task.TaskStatus == TaskStatusEnum.Pending || task.TaskStatus == TaskStatusEnum.Create) { await TenantDb.Ado.BeginTranAsync(); try { //重置任务为待处理 var result = await SetTaskStatusAsync(new TaskUpdateRequest { AutoCreateNext = false, BusinessId = request.BusinessId, BusinessType = request.BusinessType, TaskStatus = TaskStatusEnum.Create, TaskTypeName = TaskBaseTypeEnum.WAIT_ORDER_AUDIT.ToString() }, false); if (!result.Succeeded) return DataResult.Failed(result.Message, result.MultiCode); //创建&启动工作流 var result2 = await CreateAndStartWorkflow(task, false); if (!result2.Succeeded) return result2; await TenantDb.Updateable().SetColumns(x => x.TaskStatus == TaskStatusEnum.Create) .Where(x => x.BusinessId == request.BusinessId && x.BusinessType == request.BusinessType && x.TaskType == TaskBaseTypeEnum.WAIT_ORDER_AUDIT) .ExecuteCommandAsync(); await TenantDb.Ado.CommitTranAsync(); return result2; } catch (Exception ex) { await TenantDb.Ado.RollbackTranAsync(); await ex.LogAsync(Db); return DataResult.FailedWithDesc(nameof(MultiLanguageConst.Operation_Failed)); } } return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskStatusNotSupported)); } /// /// 任务审核 /// /// /// public async Task AuditAsync(AuditRequest request) { long id = request.Ids[0]; var task = await GetQuery(id, request.BusinessType.Value, TaskBaseTypeEnum.WAIT_ORDER_AUDIT).FirstAsync(); if (task == null) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.EmptyData)); if (task.TaskStatus == TaskStatusEnum.Complete) { return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskCompleted)); } else if (task.TaskStatus != TaskStatusEnum.Create && task.TaskStatus != TaskStatusEnum.Pending) { return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskAuditStatusError)); } if (task.FlowId == null) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.FlowNotFound)); var result = FlowService.Value.AuditFlowInstance(new FlowAuditInfo { AuditNote = request.Remark, Status = request.Result, Instance = await Db.Queryable().FirstAsync(x => x.Id == task.FlowId.Value) }); var flow = await Db.Queryable().Where(x => x.Id == task.FlowId.Value).Select(x => new FlowInstance { FlowStatus = x.FlowStatus, MakerList = x.MakerList, }).FirstAsync(); result.Data = new { flow.IsCompleted, flow.FlowStatus }; return result; } /// /// 创建关联任务 /// /// /// 是否使用事务 /// public async Task CreateTaskAsync(TaskCreationRequest request, bool useTransaction = true) { if (!await EnsureModuleAuthorized()) return DataResult.SuccessedWithDesc(nameof(MultiLanguageConst.ModuleUnauthorized)); var task = await GetQuery(request.BusinessId, request.BusinessType, request.TaskType).FirstAsync(); if (task != null && task.TaskStatus != TaskStatusEnum.Cancel) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskExists)); long tenatId = long.Parse(User.TenantId); string tenatName = Db.Queryable().Where(x => x.Id == tenatId).Select(x => x.Name).First(); var info = new TaskManageOrderMessageInfo { Head = new TaskManageOrderMessageHeadInfo { GID = Guid.NewGuid().ToString(), BSNO = request.BusinessId, MessageType = "WORK_FLOW_TASK", SenderId = "WorkFlow", SenderName = "工作流平台", ReceiverId = "TaskManage", ReceiverName = "任务管理平台", Version = "1.0", RequestDate = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"), RequestAction = "Add" }, Main = new TaskManageOrderMessageMainInfo { TaskType = request.TaskType, TaskSource = TaskSourceEnum.WORK_FLOW, TaskTitle = request.TaskTitle, TaskDesp = request.TaskDescription, TaskUserId = User.UserId, TaskUserName = User.UserName, TaskTenatId = tenatId, TaskTenatName = tenatName, IsCheckExistsByTaskType = true } }; if (request.RecvUserIdList == null || request.RecvUserIdList.Length == 0) { //根据配置获取默认接收人 info.Main.RecvUserInfoList = await GetRecvUsers(request.BusinessId, request.BusinessType, request.TaskType); if (info.Main.RecvUserInfoList == null || info.Main.RecvUserInfoList.Count == 0) { if (request.TaskType == TaskBaseTypeEnum.WAIT_ORDER_AUDIT) { info.Main.RecvUserInfoList = await GetRecvUsers(long.Parse(User.UserId)); } else { return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskReceiverNotFound)); } } } else { info.Main.RecvUserInfoList = await GetRecvUsers(request.RecvUserIdList); } if (info.Main.TaskTitle.IsNullOrEmpty()) { var biz = await TenantDb.Queryable().Select(x => new { x.Id, x.CustomerNo, x.MBLNO, x.Vessel, x.Voyno, x.ETD, }).FirstAsync(x => x.Id == request.BusinessId); info.Main.TaskDesp = info.Main.TaskTitle = $"【{request.TaskType.GetDescription()}】{biz?.CustomerNo} {biz?.Vessel} {biz?.Voyno} ETD:{biz?.ETD?.ToString("yyyy-MM-dd")}"; } if (useTransaction) await TenantDb.Ado.BeginTranAsync(); try { DataResult result = await ManagerService.InitTaskJob(info); if (!result.Succeeded) return result; task = new BusinessTask { BusinessId = request.BusinessId, BusinessType = request.BusinessType, TaskType = request.TaskType, TaskStatus = TaskStatusEnum.Create, RecvUsers = string.Join(',', info.Main.RecvUserInfoList.Select(x => x.RecvUserId)), CreateBy = long.Parse(User.UserId), CreateTime = DateTime.Now }; task.NextType = GetNextType(task); await TenantDb.Insertable(task).ExecuteCommandAsync(); //待审核,需创建工作流 if (request.TaskType == TaskBaseTypeEnum.WAIT_ORDER_AUDIT) { result = await CreateAndStartWorkflow(task); if (!result.Succeeded) return result; } result = await OnTaskCreated(task); if (!result.Succeeded) return result; if (useTransaction) await TenantDb.Ado.CommitTranAsync(); return result; } catch (Exception ex) { if (useTransaction) await TenantDb.Ado.RollbackTranAsync(); await ex.LogAsync(Db); return DataResult.FailedWithDesc(nameof(MultiLanguageConst.Operation_Failed)); } } /// /// 创建并启动审批工作流 /// /// /// 同时变更任务执行人 /// protected internal async Task CreateAndStartWorkflow(BusinessTask task, bool changeMarker = true) { var template = await FindTemplateAsync(AuditType.SeaExport); if (template == null) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TemplateNotFound)); var result = FlowService.Value.CreateFlowInstance(new CreateFlowInstanceReq { BusinessId = task.BusinessId, BusinessType = BusinessType.OceanShippingExport, TemplateId = template.Id }); //创建并启动实例 if (result.Succeeded) { var instance = result.Data as FlowInstance; task.FlowId = instance.Id; await TenantDb.Updateable(task).UpdateColumns(x => x.FlowId).ExecuteCommandAsync(); result = FlowService.Value.StartFlowInstance(instance.Id.ToString()); instance = result.Data as FlowInstance; if (result.Succeeded && changeMarker) { string[] ids = FlowInstanceService.GetMarkers(instance); //变更任务接收人为所有审批执行人 var users = await GetRecvUsers(ids.Select(long.Parse).ToArray()); result = await ManagerService.TransferTask(task.BusinessId, task.TaskType, users); if (result.Succeeded) { task.RecvUsers = string.Join(",", ids); await TenantDb.Updateable(task).UpdateColumns(x => x.RecvUsers).ExecuteCommandAsync(); } } } return result; } /// /// 当任务创建时调用 /// /// /// protected virtual Task OnTaskCreated(BusinessTask task) { return Task.FromResult(DataResult.Success); } /// /// 设置任务状态 /// /// /// 是否使用事务 /// public async Task> SetTaskStatusAsync(TaskUpdateRequest request, bool useTransaction = true) { if (!await EnsureModuleAuthorized()) return DataResult.SuccessMsg(MultiLanguageConst.GetDescription(nameof(MultiLanguageConst.ModuleUnauthorized)), MultiLanguageConst.ModuleUnauthorized); if (useTransaction) await TenantDb.Ado.BeginTranAsync(); try { BusinessTask task = await GetQuery(request.BusinessId, request.BusinessType, request.TaskType).FirstAsync(); if (task == null) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.EmptyData)); if (task.TaskStatus == TaskStatusEnum.Complete) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskCompleted)); if (task.TaskStatus == TaskStatusEnum.Cancel) return DataResult.FailedWithDesc(nameof(MultiLanguageConst.TaskCancelled)); var result = await ManagerService.SetTaskStatus(request.BusinessId, request.TaskType, request.TaskStatus, DateTime.Now); if (!result.Succeeded) return DataResult.Failed(result.Message, result.MultiCode); //触发任务状态变更通知 if (task.TaskStatus != request.TaskStatus) await OnTaskStatusChanged(request); //更新当前任务状态 task.TaskStatus = request.TaskStatus; if (task.TaskType == TaskBaseTypeEnum.WAIT_ORDER_AUDIT) task.FlowId = null; await TenantDb.Updateable(task).UpdateColumns(x => new { x.TaskStatus, x.FlowId }).ExecuteCommandAsync(); if (task.TaskStatus == TaskStatusEnum.Complete) { //若存在下一任务,则继续创建 if (task.NextType.HasValue && request.AutoCreateNext) { var req = new TaskCreationRequest { BusinessId = request.BusinessId, BusinessType = request.BusinessType, TaskTypeName = task.NextType.ToString(), RecvUserIdList = task.RecvUserIdArray }; result = await CreateTaskAsync(req, false); if (!result.Succeeded) return DataResult.Failed("创建下一关联任务时返回错误:" + result.Message, result.MultiCode); } } if (useTransaction) await TenantDb.Ado.CommitTranAsync(); return DataResult.Success(task.TaskStatus == TaskStatusEnum.Complete ? GetNextType(task) : null); } catch (Exception ex) { if (useTransaction) await TenantDb.Ado.RollbackTranAsync(); await ex.LogAsync(Db); return DataResult.FailedWithDesc(nameof(MultiLanguageConst.Operation_Failed)); } } /// /// 当任务状态发生变化时调用 /// /// /// protected virtual Task OnTaskStatusChanged(TaskUpdateRequest request) { return Task.CompletedTask; } /// /// 通知审批执行人变更 /// /// 回调信息 /// /// 为null时引发 public virtual async Task MarkerChangedAsync(MarkerChangedCallback callback) { ArgumentNullException.ThrowIfNull(callback, nameof(callback)); long userId = long.Parse(User.UserId); var users = await GetRecvUsers(userId); await ManagerService.SetTaskUserStatus( callback.BusinessId, TaskBaseTypeEnum.WAIT_ORDER_AUDIT, TaskStatusEnum.Complete, //callback.Status == FlowStatusEnum.Approve ? TaskStatusEnum.Complete : TaskStatusEnum.Pending DateTime.Now, users); } /// /// 审批完成回调更新 /// /// 回调信息 /// /// 为null时引发 public virtual async Task UpdateBusinessAsync(FlowCallback callback) { ArgumentNullException.ThrowIfNull(callback, nameof(callback)); //根据审批结果更新任务状态 await SetTaskStatusAsync(new TaskUpdateRequest { BusinessId = callback.BusinessId, BusinessType = callback.BusinessType.Value, TaskTypeName = TaskBaseTypeEnum.WAIT_ORDER_AUDIT.ToString(), TaskStatus = callback.FlowStatus == FlowStatusEnum.Approve ? TaskStatusEnum.Complete : TaskStatusEnum.Pending, AutoCreateNext = false //审批完成后需根据业务需要自定义任务类型,因此设置为不自动创建下一任务 }); if (callback.FlowStatus == FlowStatusEnum.Reject) { //创建审单驳回任务以进行通知 var task = await GetQuery(callback.BusinessId, callback.BusinessType.Value, TaskBaseTypeEnum.WAIT_ORDER_AUDIT).FirstAsync(); await CreateTaskAsync(new TaskCreationRequest { BusinessId = callback.BusinessId, BusinessType = callback.BusinessType.GetValueOrDefault(), TaskTypeName = TaskBaseTypeEnum.ORDER_AUDIT_REJECTED.ToString(), RecvUserIdList = [task.CreateBy] //通知任务发起人 }); } } /// /// 获取指定类型的业务关联任务 /// /// 业务ID /// 业务类型 /// 任务类型 /// protected internal ISugarQueryable GetQuery(long id, BusinessType businessType, TaskBaseTypeEnum taskType) { return TenantDb.Queryable().Where(x => x.BusinessId == id && x.BusinessType == businessType && x.TaskType == taskType); } /// /// 获取任务接收用户列表 /// /// 业务ID /// 业务类型 /// 任务类型 /// protected internal async Task?> GetRecvUsers(long id, BusinessType businessType, TaskBaseTypeEnum taskType) { string typeCode = taskType.ToString(); var allocations = await TenantDb.Queryable().Where(x => x.TaskTypeCode == typeCode) .Select(x => new { x.CarrierId, x.IsAllotCustomerService, //客服 x.IsAllotOperator, //操作 x.IsAllotSale, //销售 x.IsAllotVouchingClerk //单证 }).ToListAsync(); long? carrierId = null; switch (businessType) { case BusinessType.OceanShippingExport: carrierId = await TenantDb.Queryable().Where(x => x.Id == id).Select(x => x.CarrierId).FirstAsync(); break; } var allocation = allocations.Find(x => x.CarrierId == carrierId); //未找到匹配值 if (allocation == null) return null; var expr = Expressionable.Create(); if (allocation.IsAllotCustomerService) { expr = expr.Or(x => x.IsCustomerService); } else if (allocation.IsAllotOperator) { expr = expr.Or(x => x.IsOperator); } else if (allocation.IsAllotSale) { expr = expr.Or(x => x.IsSale); } else if (allocation.IsAllotVouchingClerk) { expr = expr.Or(x => x.IsVouchingClerk); } var condition = expr.ToExpression(); if (condition.IsNullOrEmpty()) return null; return await Db.Queryable().Where(condition).Select( x => new RecvUserInfo { RecvUserId = x.Id, RecvUserName = x.UserName }).ToListAsync(); } /// /// 获取任务接收用户列表 /// /// 用户ID /// protected internal async Task> GetRecvUsers(params long[] ids) { return await Db.Queryable().Where(x => ids.Contains(x.Id)).Select( x => new RecvUserInfo { RecvUserId = x.Id, RecvUserName = x.UserName }).ToListAsync(); } } }