using AutoMapper;
using Newtonsoft.Json;
using OASystem.Domain;
using OASystem.Domain.Dtos.Groups;
using OASystem.Domain.Entities.Groups;
using OASystem.Domain.Entities.Resource;
using System.Reflection;
namespace OASystem.Infrastructure.Repositories.Groups
{
///
/// 团组流程总览表仓储
///
public class ProcessOverviewRepository : BaseRepository
{
private readonly IMapper _mapper;
private readonly DelegationInfoRepository _groupRep;
public ProcessOverviewRepository(SqlSugarClient sqlSugar, IMapper mapper, DelegationInfoRepository groupRep) : base(sqlSugar)
{
_mapper = mapper;
_groupRep = groupRep;
}
///
/// 基础数据初始化-团组流程
///
///
///
///
public async Task> ProcessDataInitAsync(int groupId, int currUserId, List visaCountries)
{
var processs = new List();
//团组验证
var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId);
if (groupInfo == null) return processs;
// 检查是否已存在流程
var existingProcesses = await _sqlSugar.Queryable()
.Where(p => p.IsDel == 0 && p.GroupId == groupId)
.ToListAsync();
if (existingProcesses.Any()) return processs;
#region 商邀报批流程
var custInfo = await _sqlSugar.Queryable()
.Where(c => c.DiId == groupId && c.IsDel == 0)
.OrderByDescending(c => c.CreateTime)
.FirstAsync();
string oaNode2Tips = "客户提供完整名单后,2周内取得邀请函(翻译件)。";
if (custInfo != null)
{
oaNode2Tips = $"请于{custInfo.CreateTime.AddDays(14):yyyy年MM月dd日}内完成该项工作(客户提供完整名单后,2周内取得邀请函(翻译件)。)";
}
var oaNode4Tips = $"请于{groupInfo.VisitDate.AddDays(-5):yyyy年MM月dd日}内完成该项工作(按进度实际公务活动落实情况,出发前5日落实公务。)";
processs.Add(
Grp_ProcessOverview.Create(groupId, 1, GroupProcessType.Invitation, ProcessStatus.InProgress, currUserId,
new List()
{
Grp_ProcessNode.Create(1, "报批基础资料准备","更新报批行程和请示,提供其他报批所需材料,4个工作日内完成。",ProcessStatus.InProgress, true,currUserId),
Grp_ProcessNode.Create(2, "报批邀请函资料准备",oaNode2Tips, ProcessStatus.InProgress, false,currUserId),
Grp_ProcessNode.Create(3, "获得批件","提供完整的报批全套资源。",ProcessStatus.InProgress, false, currUserId ),
Grp_ProcessNode.Create(4, "对接公务",oaNode4Tips,ProcessStatus.InProgress, false, currUserId),
Grp_ProcessNode.Create(5, "参与翻译对接","",ProcessStatus.InProgress, false, currUserId),
Grp_ProcessNode.Create(6, "商邀文案配合","",ProcessStatus.InProgress, false, currUserId),
}));
#endregion
#region 签证流程
//单独处理签证流程节点
var visaNodes = new List();
if (visaCountries != null && visaCountries.Count > 0)
{
var visaDefualtNodes = new List();
for (int i = 1; i < visaCountries.Count + 1; i++)
{
visaDefualtNodes.Add(VisaProcessNode.Info(i, visaCountries[i - 1].ToString()));
}
visaNodes.Add(Grp_ProcessNode.Create(1, "签证", "", ProcessStatus.InProgress, true, currUserId, JsonConvert.SerializeObject(visaDefualtNodes)));
}
processs.Add(Grp_ProcessOverview.Create(groupId, 2, GroupProcessType.Visa, ProcessStatus.InProgress, currUserId, visaNodes));
#endregion
#region 机票流程
string airNode1Tips = "建团后打勾确认出团的时候开始24小时内。";
if (groupInfo.Step == 1 || groupInfo.Step == 2)
{
if (groupInfo.StepOperationTime.HasValue)
{
airNode1Tips = $"请于{groupInfo.StepOperationTime.Value.AddDays(1):yyyy年MM月dd日}内完成该项工作(建团后打勾确认出团的时候开始24小时内)";
}
}
var airNode5Tips = $"请于{groupInfo.VisitDate.AddDays(-5):yyyy年MM月dd日}内完成该项工作(团组出发前5日)";
processs.Add(
Grp_ProcessOverview.Create(groupId, 3, GroupProcessType.AirTicket, ProcessStatus.InProgress, currUserId,
new List()
{
Grp_ProcessNode.Create(1, "初步拟定航程方案及价格", airNode1Tips, ProcessStatus.InProgress, true,currUserId ),
Grp_ProcessNode.Create(2, "机票占位、续位", "", ProcessStatus.UnStarted, false,currUserId ),
Grp_ProcessNode.Create(3, "完成机票采购确认(含预算核对、出票确认等)", "", ProcessStatus.UnStarted,false,currUserId),
Grp_ProcessNode.Create(4, "进行出票操作并核查信息", "", ProcessStatus.UnStarted, false, currUserId),
Grp_ProcessNode.Create(5, "机票已出", airNode5Tips, ProcessStatus.UnStarted, false, currUserId),
Grp_ProcessNode.Create(6, "完成机票选座", "", ProcessStatus.UnStarted, false,currUserId)
}
)
);
#endregion
#region 酒店流程
string hotelNode1Tips = "建团后打勾确认出团的时候开始2个工作日。";
if (groupInfo.Step == 1 || groupInfo.Step == 2)
{
if (groupInfo.StepOperationTime.HasValue)
{
hotelNode1Tips = $"请于{groupInfo.StepOperationTime.Value.AddDays(2):yyyy年MM月dd日}内完成该项工作(建团后打勾确认出团的时候开始2个工作日)";
}
}
var hotelNode4Tips = $"请于{groupInfo.VisitDate.AddDays(-5):yyyy年MM月dd日}内完成该项工作(团组出发前5天)";
var hotelNode5Tips = $"请于{groupInfo.VisitEndDate.AddDays(5):yyyy年MM月dd日}内完成该项工作(团组结束后5天内)";
processs.Add(
Grp_ProcessOverview.Create(groupId, 4, GroupProcessType.Hotel, ProcessStatus.InProgress, currUserId,
new List()
{
Grp_ProcessNode.Create(1, "筛选并按照预算标准,对目标酒店进行询价、比价、谈价", hotelNode1Tips, ProcessStatus.InProgress, true, currUserId),
Grp_ProcessNode.Create(2, "获取酒店确认函与入住名单核对", "", ProcessStatus.UnStarted, false, currUserId ),
Grp_ProcessNode.Create(3, "预订酒店并录入OA", "", ProcessStatus.UnStarted,false,currUserId ),
Grp_ProcessNode.Create(4, "行前再次确认酒店订单、付款状态及入住安排", hotelNode4Tips,ProcessStatus.UnStarted, false,currUserId ),
Grp_ProcessNode.Create(5, "行程结束后整理酒店发票与结算", hotelNode5Tips, ProcessStatus.UnStarted, false, currUserId ),
}
)
);
#endregion
#region 地接流程
var airTripCodeInfo = await _sqlSugar.Queryable()
.Where(x => x.IsDel == 0 && x.DiId == groupId)
.OrderByDescending(x => x.CreateTime)
.FirstAsync();
string opNode1Tips = $"机票行程代码最后一段录入后1个工作日内。";
if (airTripCodeInfo != null)
{
opNode1Tips = $"请于{airTripCodeInfo.CreateTime.AddDays(1):yyyy年MM月dd日}内完成该项工作(机票行程代码最后一段录入后1个工作日内)";
}
string opNode2Tips = $"请于{groupInfo.CreateTime.AddDays(7):yyyy年MM月dd日}内完成该项工作(建团完成后7个工作日内)";
string opNode3Tips = $"请于{groupInfo.CreateTime.AddDays(10):yyyy年MM月dd日}内完成该项工作(上一步往后3个工作日内)";
string opNode4Tips = $"请于{groupInfo.CreateTime.AddDays(12):yyyy年MM月dd日}内完成该项工作(上一步往后2个工作日内)";
var backListInfo = await _sqlSugar.Queryable().Where(x => x.DiId == groupId && x.IsDel == 0).FirstAsync();
string opNode5Tips = $"倒推表里开行前会 -3天。";
if (backListInfo != null) {
if (DateTime.TryParse(backListInfo.PreTripMeetingDt,out DateTime dateTime))
{
opNode5Tips = $"请于{dateTime.AddDays(-3):yyyy年MM月dd日}内完成该项工作(倒推表里开行前会 -3天)";
}
}
processs.Add(
Grp_ProcessOverview.Create(groupId, 5, GroupProcessType.LocalGuide, ProcessStatus.InProgress, currUserId,
new List()
{
Grp_ProcessNode.Create(1,"根据机票方案出框架行程", opNode1Tips,ProcessStatus.InProgress, true, currUserId ),
Grp_ProcessNode.Create(2,"联系并询价地接、餐厅、用车、景点等供应商", opNode2Tips,ProcessStatus.UnStarted, false, currUserId ),
Grp_ProcessNode.Create(3,"提交供应商报价及比价表", opNode3Tips, ProcessStatus.UnStarted, false, currUserId),
Grp_ProcessNode.Create(4,"执行采购流程", opNode4Tips, ProcessStatus.UnStarted, false, currUserId),
Grp_ProcessNode.Create(5,"制定最终《行程表》及《出行手册》", opNode5Tips, ProcessStatus.UnStarted, false, currUserId ),
Grp_ProcessNode.Create(6,"送机", "", ProcessStatus.UnStarted, false, currUserId ),
}
)
);
#endregion
#region 费用结算流程
processs.Add(
Grp_ProcessOverview.Create(groupId, 6, GroupProcessType.FeeSettle, ProcessStatus.InProgress, currUserId,
new List()
{
Grp_ProcessNode.Create(1, "费用结算中", "", ProcessStatus.InProgress, true,currUserId ),
Grp_ProcessNode.Create(2, "费用结算完毕", "", ProcessStatus.UnStarted, false,currUserId ),
}
)
);
#endregion
return processs;
}
///
/// 团组流程初始化
///
/// 创建流程请求参数
/// 创建的流程信息
public async Task ProcessInitAsync(int groupId, int currUserId)
{
//团组验证
var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId);
if (groupInfo == null)
{
return new Result { Code = 400, Msg = "团组不存在" };
}
// 检查是否已存在流程
var existingProcesses = await _sqlSugar.Queryable()
.Where(p => p.IsDel == 0 && p.GroupId == groupId)
.ToListAsync();
if (existingProcesses.Any())
{
return new Result { Code = 400, Msg = "该团组的流程已存在" };
}
//处理签证国家
var visaCountries = _groupRep.GroupSplitCountry(groupInfo.VisitCountry);
// 定义默认的流程节点
var processs = await ProcessDataInitAsync(groupId, currUserId, visaCountries);
_sqlSugar.BeginTran();
foreach (var item in processs)
{
var processId = await _sqlSugar.Insertable(item).ExecuteReturnIdentityAsync();
if (processId < 1)
{
_sqlSugar.RollbackTran();
return new Result { Code = 400, Msg = "团组流程进度总览表添加失败!" };
}
// 记录流程日志
await LogProcessOpAsync(null, item, "Create", currUserId);
var nodes = item.Nodes.Select((nodeDto, index) => new Grp_ProcessNode
{
ProcessId = processId,
NodeName = nodeDto.NodeName,
NodeOrder = nodeDto.NodeOrder,
OverallStatus = nodeDto.OverallStatus,
NodeDescTips = nodeDto.NodeDescTips,
//Country = nodeDto.Country,
IsCurrent = nodeDto.IsCurrent,
Remark = nodeDto.Remark
}).ToList();
var nodeIds = await _sqlSugar.Insertable(nodes).ExecuteCommandAsync();
if (nodeIds < 1)
{
_sqlSugar.RollbackTran();
return new Result { Code = 400, Msg = "团组流程进度流程节点添加失败!" };
}
//记录节点日志
foreach (var node in nodes)
{
await LogNodeOpAsync(null, node, "Create", currUserId);
}
}
_sqlSugar.CommitTran();
return new Result { Code = 200, Msg = "添加成功!" }; ;
}
///
/// 获取团组的所有流程及流程详情
///
/// 创建流程请求参数
/// 创建的流程信息
public async Task ProcessesDetailsAsync(int groupId)
{
//团组验证
var groupInfo = await _sqlSugar.Queryable().FirstAsync(g => g.Id == groupId);
if (groupInfo == null)
{
return new Result { Code = 400, Msg = "团组不存在" };
}
// 检查是否已存在流程
var existingProcesses = await _sqlSugar.Queryable().Where(p => p.IsDel == 0 && p.GroupId == groupId).ToListAsync();
if (!existingProcesses.Any())
{
//新建团组流程
var res = await ProcessInitAsync(groupId, 4);
if (res.Code != 200)
{
return res;
}
}
var users = await _sqlSugar.Queryable().Select(x => new {x.Id,x.CnName }).ToListAsync();
var processData = await _sqlSugar.Queryable()
.Where(p => p.GroupId == groupId && p.IsDel == 0)
.Mapper(p => p.Nodes, p => p.Nodes.First().ProcessId)
.ToListAsync();
var processes = processData.Select(p => new
{
p.Id,
p.GroupId,
p.ProcessType,
ProcessName = p.ProcessType.GetEnumDescription(),
//p.OverallStatus,
//StatusText = p.OverallStatus.GetDescription(),
Nodes = p.Nodes.Select(n =>
{
//单独处理签证板块
var visaSubNodes = new List();
string remark = string.Empty;
if (p.ProcessType == GroupProcessType.Visa)
{
visaSubNodes = JsonConvert.DeserializeObject>(n.Remark);
}
return new
{
n.Id,
n.ProcessId,
n.NodeOrder,
n.NodeName,
n.OverallStatus,
StatusText = n.OverallStatus.GetEnumDescription(),
Operator = users.FirstOrDefault(u => u.Id == n.Operator)?.CnName ?? "-",
OpeateTime = n.OperationTime.HasValue ? n.OperationTime.Value.ToString("yyyy-MM-dd HH:mm:ss") : "-",
ActualDone = n.ActualDone?.ToString("yyyy-MM-dd HH:mm:ss") ?? "",
n.NodeDescTips,
//节点类型为签证时使用
visaSubNodes
};
}).OrderBy(n => n.NodeOrder).ToList()
}).ToList();
return new Result { Code = 200, Data = processes, Msg = "查询成功!" };
}
///
/// 更新节点状态
///
/// 节点ID
/// 当前用户ID
/// 流程状态,默认为已完成
/// 操作结果
public async Task UpdateNodeStatusAsync(int nodeId, int currUserId, ProcessStatus processStatus = ProcessStatus.Completed)
{
try
{
// 使用事务确保数据一致性
var result = await _sqlSugar.Ado.UseTranAsync(async () =>
{
// 1. 获取并验证节点
var node = await _sqlSugar.Queryable()
.FirstAsync(n => n.Id == nodeId && n.IsDel == 0);
if (node == null)
{
throw new BusinessException("当前节点不存在或已被删除。");
}
// 2. 获取流程信息,检查ProcessType
var process = await _sqlSugar.Queryable()
.FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0);
if (process == null)
{
throw new BusinessException("关联的流程不存在。");
}
// 3. 节点操作验证
ValidateNodeOperation(node, processStatus);
// 4. 存储更新前的值
var before = new Grp_ProcessNode()
{
Id = node.Id,
ProcessId = node.ProcessId,
NodeName = node.NodeName,
NodeOrder = node.NodeOrder,
OverallStatus = node.OverallStatus,
Operator = node.Operator,
OperationTime = node.OperationTime,
IsCurrent = node.IsCurrent,
};
// 5. 更新节点状态
node.OverallStatus = processStatus;
node.Operator = currUserId;
node.OperationTime = DateTime.Now;
var updateCount = await _sqlSugar.Updateable(node)
.UpdateColumns(n => new
{
n.OverallStatus,
n.Operator,
n.OperationTime
})
.ExecuteCommandAsync();
if (updateCount == 0)
{
throw new BusinessException("节点状态更新失败。");
}
// 6. 记录节点日志
await LogNodeOpAsync(before, node, "Update", currUserId);
// 7. 如果是完成当前节点,处理流程流转
// 当前节点或者流程类型为商邀可进入状态流转
if (processStatus == ProcessStatus.Completed && (node.IsCurrent || process.ProcessType == GroupProcessType.Invitation))
{
await ProcessCurrentNodeCompletionAsync(node, currUserId);
}
return new Result { Code = StatusCodes.Status200OK, Msg = "操作成功。" };
});
return result.IsSuccess ? result.Data : new Result
{
Code = StatusCodes.Status500InternalServerError,
Msg = result.ErrorMessage
};
}
catch (BusinessException ex)
{
// 业务异常
return new Result { Code = StatusCodes.Status400BadRequest, Msg = ex.Message };
}
catch (Exception ex)
{
// 系统异常
return new Result { Code = StatusCodes.Status500InternalServerError, Msg = "系统错误,请稍后重试" };
}
}
///
/// 验证节点操作权限
///
/// 流程节点
/// 目标状态
private static void ValidateNodeOperation(Grp_ProcessNode node, ProcessStatus targetStatus)
{
// 验证节点是否已完成
if (node.OverallStatus == ProcessStatus.Completed)
{
throw new BusinessException("当前节点已完成,不可重复操作。");
}
// 验证状态流转是否合法(可选)
//if (targetStatus != ProcessStatus.Completed)
//{
// throw new BusinessException("未开始或者进行中的节点只能重新完成,不可进行其他操作。");
//}
// 验证是否尝试将已完成节点改为其他状态
if (node.OverallStatus == ProcessStatus.Completed && targetStatus != ProcessStatus.Completed)
{
throw new BusinessException("已完成节点不可修改状态。");
}
}
///
/// 处理当前节点完成后的流程流转
///
private async Task ProcessCurrentNodeCompletionAsync(Grp_ProcessNode currentNode, int currUserId)
{
// 1. 获取流程信息
var process = await _sqlSugar.Queryable()
.FirstAsync(p => p.Id == currentNode.ProcessId && p.IsDel == 0);
if (process == null)
{
throw new BusinessException("关联的流程不存在。");
}
var processBefore = new Grp_ProcessOverview()
{
Id = process.Id,
GroupId = process.GroupId,
ProcessOrder = process.ProcessOrder,
ProcessType = process.ProcessType,
OverallStatus = process.OverallStatus,
StartTime = process.StartTime,
EndTime = process.EndTime,
UpdatedUserId = process.UpdatedUserId,
UpdatedTime = process.UpdatedTime
};
// 2. 取消当前节点的当前状态
var before = new Grp_ProcessNode()
{
Id = currentNode.Id,
ProcessId = currentNode.ProcessId,
NodeName = currentNode.NodeName,
NodeOrder = currentNode.NodeOrder,
OverallStatus = currentNode.OverallStatus,
Operator = currentNode.Operator,
OperationTime = currentNode.OperationTime,
IsCurrent = currentNode.IsCurrent,
};
currentNode.IsCurrent = false;
await _sqlSugar.Updateable(currentNode)
.UpdateColumns(n => new { n.IsCurrent })
.ExecuteCommandAsync();
// 2.1 记录节点日志 取消当前节点状态
await LogNodeOpAsync(before, currentNode, "Update", currUserId);
// 3. 查找并激活下一个节点 商邀节点单独处理
if (process.ProcessType == GroupProcessType.Invitation)
{
var invitaNodeStatus = await _sqlSugar.Queryable()
.Where(x => x.IsDel == 0 && x.ProcessId == currentNode.ProcessId)
.ToListAsync();
int completedCount = invitaNodeStatus.Count(n => n.OverallStatus == ProcessStatus.Completed);
int nodeCount = invitaNodeStatus.Count;
if (completedCount == nodeCount) //全部子节点完成,该流程完成
{
process.OverallStatus = ProcessStatus.Completed;
process.EndTime = DateTime.Now;
}
}
else
{
var nextNode = await _sqlSugar.Queryable()
.Where(n => n.ProcessId == currentNode.ProcessId
&& n.NodeOrder == currentNode.NodeOrder + 1
&& n.IsDel == 0)
.FirstAsync();
if (nextNode != null)
{
var nextNodeBefore = new Grp_ProcessNode()
{
Id = nextNode.Id,
ProcessId = nextNode.ProcessId,
NodeName = nextNode.NodeName,
NodeOrder = nextNode.NodeOrder,
OverallStatus = nextNode.OverallStatus,
Operator = nextNode.Operator,
OperationTime = nextNode.OperationTime,
IsCurrent = nextNode.IsCurrent,
};
// 激活下一个节点
nextNode.IsCurrent = true;
nextNode.OverallStatus = ProcessStatus.InProgress;
//nextNode.Operator = currUserId;
//nextNode.OperationTime = DateTime.Now;
var updateCount = await _sqlSugar.Updateable(nextNode)
.UpdateColumns(n => new
{
n.IsCurrent,
n.OverallStatus,
n.Operator,
n.OperationTime
})
.ExecuteCommandAsync();
if (updateCount == 0)
{
throw new BusinessException("激活下一节点失败");
}
// 1.1 记录节点日志 激活下一节点当前节点状态
await LogNodeOpAsync(nextNodeBefore, nextNode, "Start", currUserId);
// 更新流程状态为进行中
process.OverallStatus = ProcessStatus.InProgress;
}
else
{
// 下一节点不存在,整个流程完成
process.OverallStatus = ProcessStatus.Completed;
process.EndTime = DateTime.Now;
}
}
// 4. 更新流程信息
process.UpdatedUserId = currUserId;
process.UpdatedTime = DateTime.Now;
var processUpdateCount = await _sqlSugar.Updateable(process)
.UpdateColumns(p => new
{
p.OverallStatus,
p.EndTime,
p.UpdatedUserId,
p.UpdatedTime
})
.ExecuteCommandAsync();
if (processUpdateCount == 0)
{
throw new BusinessException("流程状态更新失败。");
}
//记录流程日志
await LogProcessOpAsync(processBefore, process, "Update", currUserId);
}
///
/// 更新签证节点信息及状态
///
/// 签证节点更新数据传输对象
/// 操作结果
public async Task UpdateVisaNodeDetailsAsync(GroupProcessUpdateVisaNodeDetailsDto dto)
{
// 1. 获取并验证节点和流程
var node = await _sqlSugar.Queryable()
.FirstAsync(n => n.Id == dto.NodeId && n.IsDel == 0)
?? throw new BusinessException("当前节点不存在或已被删除。");
var process = await _sqlSugar.Queryable()
.FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0)
?? throw new BusinessException("当前流程不存在或已被删除。");
if (process.ProcessType != GroupProcessType.Visa)
{
throw new BusinessException("当前流程节点不为签证流程,不可编辑。");
}
// 2. 检查签证子节点 字段信息是否全部填写
var allSubNodesCompleted = dto.VisaSubNodes?.All(subNode => EntityExtensions.IsCompleted(subNode)) ?? false;
// 2.1 存储更新前流程及节点信息
var nodeBefore = new Grp_ProcessNode()
{
Id = node.Id,
ProcessId = node.ProcessId,
NodeName = node.NodeName,
NodeOrder = node.NodeOrder,
OverallStatus = node.OverallStatus,
Operator = node.Operator,
OperationTime = node.OperationTime,
IsCurrent = node.IsCurrent,
};
var processBefore = new Grp_ProcessOverview()
{
Id = process.Id,
GroupId = process.GroupId,
ProcessOrder = process.ProcessOrder,
ProcessType = process.ProcessType,
OverallStatus = process.OverallStatus,
StartTime = process.StartTime,
EndTime = process.EndTime,
UpdatedUserId = process.UpdatedUserId,
UpdatedTime = process.UpdatedTime
};
// 3. 更新节点信息
node.Remark = JsonConvert.SerializeObject(dto.VisaSubNodes);
node.Operator = dto.CurrUserId;
node.OperationTime = DateTime.Now;
if (allSubNodesCompleted)
{
node.OverallStatus = ProcessStatus.Completed;
process.OverallStatus = ProcessStatus.Completed;
process.EndTime = DateTime.Now;
process.UpdatedUserId = dto.CurrUserId;
process.UpdatedTime = DateTime.Now;
// 更新流程状态
await _sqlSugar.Updateable(process)
.UpdateColumns(p => new
{
p.OverallStatus,
p.EndTime,
p.UpdatedUserId,
p.UpdatedTime
})
.ExecuteCommandAsync();
//记录流程日志
await LogProcessOpAsync(processBefore, process, "Update", dto.CurrUserId);
}
// 4. 保存节点更新
await _sqlSugar.Updateable(node)
.UpdateColumns(n => new
{
n.Remark,
n.Operator,
n.OperationTime,
n.OverallStatus
})
.ExecuteCommandAsync();
//记录节点日志
await LogNodeOpAsync(nodeBefore, node, "Update", dto.CurrUserId);
return new Result { Code = 200, Msg = "节点信息更新成功。" };
}
///
/// 更新签证节点信息及状态
///
/// 签证节点更新数据传输对象
/// 操作结果
public async Task SetActualDoneAsync(int nodeId, DateTime dt, int currUserId)
{
// 1. 获取并验证节点和流程
var node = await _sqlSugar.Queryable()
.FirstAsync(n => n.Id == nodeId && n.IsDel == 0)
?? throw new BusinessException("当前节点不存在或已被删除。");
var process = await _sqlSugar.Queryable()
.FirstAsync(p => p.Id == node.ProcessId && p.IsDel == 0)
?? throw new BusinessException("当前流程不存在或已被删除。");
// 2.1 存储更新前流程及节点信息
var nodeBefore = new Grp_ProcessNode()
{
Id = node.Id,
ProcessId = node.ProcessId,
NodeName = node.NodeName,
NodeOrder = node.NodeOrder,
OverallStatus = node.OverallStatus,
Operator = node.Operator,
OperationTime = node.OperationTime,
IsCurrent = node.IsCurrent,
};
node.ActualDone = dt;
// 3. 保存节点更新
await _sqlSugar.Updateable(node)
.UpdateColumns(n => new
{
n.ActualDone
})
.ExecuteCommandAsync();
//记录节点日志
await LogNodeOpAsync(nodeBefore, node, "Update", currUserId);
return new Result { Code = 200, Msg = "实际操作时间设置成功。" };
}
#region 操作日志
///
/// 记录流程操作日志
///
/// 操作前
/// 操作后
/// 操作类型(Create - 创建、Update - 更新、Complete - 完成)
/// 操作人ID
/// 异步任务
public async Task LogProcessOpAsync(Grp_ProcessOverview before, Grp_ProcessOverview after,string opType, int operId)
{
var chgDetails = GetProcessChgDetails(before, after);
var log = new Grp_ProcessLog
{
ProcessId = after?.Id ?? before?.Id,
GroupId = after?.GroupId ?? before?.GroupId ?? 0,
OpType = opType,
OpDesc = GenerateProcessOpDesc(opType, before, after, chgDetails),
BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null,
AfterData = after != null ? JsonConvert.SerializeObject(after, GetJsonSettings()) : null,
ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)),
CreateUserId = operId
};
await _sqlSugar.Insertable(log).ExecuteCommandAsync();
}
///
/// 记录节点操作日志
///
/// 操作前
/// 操作后
/// 操作类型(Create - 创建、Update - 更新、Start - 启动、Complete - 完成)
/// 操作人ID
/// 异步任务
public async Task LogNodeOpAsync(Grp_ProcessNode before, Grp_ProcessNode after,string opType, int operId)
{
var chgDetails = GetNodeChgDetails(before, after);
var log = new Grp_ProcessLog
{
NodeId = after?.Id ?? before?.Id,
ProcessId = after?.ProcessId ?? before?.ProcessId,
GroupId = 0, // 通过流程ID关联获取
OpType = opType,
OpDesc = GenerateNodeOpDesc(opType, before, after, chgDetails),
BeforeData = before != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null,
AfterData = after != null ? JsonConvert.SerializeObject(before, GetJsonSettings()) : null,
ChgFields = string.Join(",", chgDetails.Select(x => x.FieldName)),
CreateUserId = operId
};
await _sqlSugar.Insertable(log).ExecuteCommandAsync();
}
///
/// 获取流程变更详情
///
/// 变更前
/// 变更后
/// 变更详情
private List GetProcessChgDetails(Grp_ProcessOverview before, Grp_ProcessOverview after)
{
var chgDetails = new List();
if (before == null || after == null) return chgDetails;
var props = typeof(Grp_ProcessOverview).GetProperties(BindingFlags.Public | BindingFlags.Instance)
.Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name));
foreach (var prop in props)
{
var beforeVal = prop.GetValue(before);
var afterVal = prop.GetValue(after);
if (!Equals(beforeVal, afterVal))
{
chgDetails.Add(new FieldChgDetail
{
FieldName = prop.Name,
BeforeValue = FormatVal(beforeVal),
AfterValue = FormatVal(afterVal)
});
}
}
return chgDetails;
}
///
/// 获取节点变更详情
///
/// 变更前
/// 变更后
/// 变更详情
private List GetNodeChgDetails(Grp_ProcessNode before, Grp_ProcessNode after)
{
var chgDetails = new List();
if (before == null || after == null) return chgDetails;
var props = typeof(Grp_ProcessNode).GetProperties(BindingFlags.Public | BindingFlags.Instance)
.Where(p => p.CanRead && p.CanWrite && !IsExclField(p.Name));
foreach (var prop in props)
{
var beforeVal = prop.GetValue(before);
var afterVal = prop.GetValue(after);
if (!Equals(beforeVal, afterVal))
{
chgDetails.Add(new FieldChgDetail
{
FieldName = prop.Name,
BeforeValue = FormatVal(beforeVal),
AfterValue = FormatVal(afterVal)
});
}
}
return chgDetails;
}
///
/// 生成流程操作描述
///
/// 操作类型
/// 操作前
/// 操作后
/// 变更详情
/// 操作描述
private string GenerateProcessOpDesc(string opType, Grp_ProcessOverview before,
Grp_ProcessOverview after, List chgDetails)
{
var processType = after?.ProcessType ?? before?.ProcessType;
var processName = GetProcessTypeName(processType);
if (!chgDetails.Any())
{
return opType switch
{
"Create" => $"创建流程:{processName}",
"Update" => $"更新流程:{processName} - 无变更",
//"Start" => $"启动流程:{processName}",
"Complete" => $"完成流程:{processName}",
//"Delete" => $"删除流程:{processName}",
_ => $"{opType}:{processName}"
};
}
var chgDesc = string.Join("; ", chgDetails.Select(x =>
$"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})"));
return $"{GetOpTypeDisplay(opType)}:{processName} - {chgDesc}";
}
///
/// 获取JSON序列化设置
///
/// JSON设置
private static JsonSerializerSettings GetJsonSettings()
{
return new JsonSerializerSettings
{
ReferenceLoopHandling = ReferenceLoopHandling.Ignore,
NullValueHandling = NullValueHandling.Ignore,
DateFormatString = "yyyy-MM-dd HH:mm:ss",
Formatting = Formatting.None
};
}
///
/// 生成节点操作描述
///
/// 操作类型
/// 操作前
/// 操作后
/// 变更详情
/// 操作描述
private string GenerateNodeOpDesc(string opType, Grp_ProcessNode before,
Grp_ProcessNode after, List chgDetails)
{
var nodeName = after?.NodeName ?? before?.NodeName;
if (!chgDetails.Any())
{
return opType switch
{
"Create" => $"创建节点:{nodeName}",
"Update" => $"更新节点:{nodeName} - 无变更",
"Start" => $"启动节点:{nodeName}",
"Complete" => $"完成节点:{nodeName}",
//"Delete" => $"删除节点:{nodeName}",
_ => $"{opType}:{nodeName}"
};
}
var chgDesc = string.Join("; ", chgDetails.Select(x =>
$"{GetFieldDisplayName(x.FieldName)} ({x.BeforeValue} -> {x.AfterValue})"));
return $"{GetOpTypeDisplay(opType)}:{nodeName} - {chgDesc}";
}
///
/// 获取流程类型名称
///
/// 流程类型
/// 流程名称
private static string GetProcessTypeName(GroupProcessType? processType)
{
return processType switch
{
GroupProcessType.Invitation => "商邀报批",
GroupProcessType.Visa => "签证",
GroupProcessType.AirTicket => "机票",
GroupProcessType.Hotel => "酒店",
GroupProcessType.LocalGuide => "地接",
GroupProcessType.FeeSettle => "费用结算",
_ => "未知流程"
};
}
///
/// 获取操作类型显示
///
/// 操作类型
/// 显示名称
private static string GetOpTypeDisplay(string opType)
{
return opType switch
{
"Create" => "创建",
"Update" => "更新",
"Start" => "启动",
"Complete" => "完成",
"Delete" => "删除",
"StatusChg" => "状态变更",
_ => opType
};
}
///
/// 获取字段显示名称
///
/// 字段名
/// 显示名称
private string GetFieldDisplayName(string fieldName)
{
return fieldName switch
{
"OverallStatus" => "状态",
"ProcessOrder" => "流程顺序",
"StartTime" => "开始时间",
"EndTime" => "结束时间",
"NodeOrder" => "节点顺序",
"NodeName" => "节点名称",
"IsCurrent" => "当前节点",
"Operator" => "操作人",
"OperationTime" => "操作时间",
_ => fieldName
};
}
///
/// 格式化值显示
///
/// 值
/// 格式化值
private string FormatVal(object value)
{
if (value == null) return "空";
if (value is ProcessStatus status)
{
return status switch
{
ProcessStatus.UnStarted => "未开始",
ProcessStatus.InProgress => "进行中",
ProcessStatus.Completed => "已完成",
_ => status.ToString()
};
}
if (value is bool boolVal) return boolVal ? "是" : "否";
if (value is DateTime dateVal) return dateVal.ToString("yyyy-MM-dd HH:mm");
var strVal = value.ToString();
return string.IsNullOrEmpty(strVal) ? "空" : strVal;
}
///
/// 检查是否排除字段
///
/// 字段名
/// 是否排除
private bool IsExclField(string fieldName)
{
var exclFields = new List
{
"Id", "CreateTime", "CreateUserId", "UpdatedTime", "UpdatedUserId",
"Nodes", "Process" // 导航属性
};
return exclFields.Contains(fieldName);
}
///
/// 获取流程日志
///
/// 流程ID
/// 日志列表
public async Task> GetProcessLogsAsync(int processId)
{
return await _sqlSugar.Queryable()
.Where(x => x.ProcessId == processId)
.OrderByDescending(x => x.CreateTime)
.ToListAsync();
}
///
/// 获取团组流程日志
///
/// 团组ID
/// 日志列表
public async Task> GetGroupLogsAsync(int groupId)
{
return await _sqlSugar.Queryable()
.Where(x => x.GroupId == groupId)
.OrderByDescending(x => x.CreateTime)
.ToListAsync();
}
#endregion
}
}