// 大名科技(天津)有限公司 版权所有 // // 此源代码遵循位于源代码树根目录中的 LICENSE 文件的许可证 // // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动 // // 任何基于本项目二次开发而产生的一切法律纠纷和责任,均与作者无关 namespace Admin.NET.Plugin.DingTalk.Service; /// /// 钉钉服务 🧩 /// [ApiDescriptionSettings(DingTalkConst.GroupName, Module = "DingTalk", Order = 100)] public class DingTalkService : IDynamicApiController, IScoped { private readonly IDingTalkApi _dingTalkApi; private readonly DingTalkOptions _dingTalkOptions; private readonly SqlSugarRepository _dingTalkUserRepo; private readonly SqlSugarRepository _sysUserRep; public DingTalkService(IDingTalkApi dingTalkApi, IOptions dingTalkOptions, SqlSugarRepository dingTalkUserRepo, SqlSugarRepository sysUserRep) { _dingTalkApi = dingTalkApi; _dingTalkOptions = dingTalkOptions.Value; _dingTalkUserRepo = dingTalkUserRepo; _sysUserRep = sysUserRep; } /// /// 同步钉钉用户 🔖 /// /// [DisplayName("同步钉钉用户")] public async Task SyncDingTalkUser() { var param = new GetDingTalkTokenInput() { AppKey = _dingTalkOptions.ClientId, AppSecret = _dingTalkOptions.ClientSecret }; var tokenRes = await _dingTalkApi.GetDingTalkToken(param); if (tokenRes.ErrCode != 0) throw Oops.Oh(tokenRes.ErrMsg); var offset = 0; while (offset >= 0) { // 获取用户Id列表 var userIdsRes = await _dingTalkApi.GetDingTalkCurrentEmployeesList(tokenRes.AccessToken, new GetDingTalkCurrentEmployeesListInput { StatusList = "2,3,5,-1", Size = 50, Offset = offset }); if (!userIdsRes.Success) throw Oops.Oh(userIdsRes.ErrMsg); // 根据用户Id获取花名册 var rosterRes = await _dingTalkApi.GetDingTalkCurrentEmployeesRosterList(tokenRes.AccessToken, new GetDingTalkCurrentEmployeesRosterListInput() { UserIdList = string.Join(",", userIdsRes.Result.DataList), FieldFilterList = $"{DingTalkConst.NameField},{DingTalkConst.JobNumberField},{DingTalkConst.MobileField}", AgentId = _dingTalkOptions.AgentId }); if (!rosterRes.Success) throw Oops.Oh(rosterRes.ErrMsg); // 判断新增还是更新 var userIds = rosterRes.Result.Select(u => u.UserId).ToList(); var uDingTalkUser = await _dingTalkUserRepo.AsQueryable() .Where(u => userIds.Contains(u.DingTalkUserId)) .ToListAsync(); var uUserIds = uDingTalkUser.Select(u => u.DingTalkUserId); // 需要更新的用户Id var iUserIds = userIds.Where(u => !uUserIds.Contains(u)); // 需要新增的用户Id // 保存钉钉用户 var iUsers = rosterRes.Result .Where(u => iUserIds.Contains(u.UserId)) .Select(u => new DingTalkUser { DingTalkUserId = u.UserId, Name = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.NameField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), Mobile = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.MobileField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), JobNumber = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.JobNumberField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), }).ToList(); if (iUsers.Count > 0) { await _dingTalkUserRepo.AsInsertable(iUsers).ExecuteCommandAsync(); } // 更新钉钉用户 var uUsers = rosterRes.Result .Where(u => uUserIds.Contains(u.UserId)) .Select(u => new DingTalkUser { Id = uDingTalkUser.Where(m => m.DingTalkUserId == u.UserId).Select(m => m.Id).FirstOrDefault(), DingTalkUserId = u.UserId, Name = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.NameField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), Mobile = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.MobileField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), JobNumber = u.FieldDataList.Where(m => m.FieldCode == DingTalkConst.JobNumberField).Select(m => m.FieldValueList.Select(n => n.Value).FirstOrDefault()).FirstOrDefault(), }).ToList(); if (uUsers.Count > 0) { await _dingTalkUserRepo.AsUpdateable(uUsers).UpdateColumns(u => new { u.DingTalkUserId, u.Name, u.Mobile, u.JobNumber, u.UpdateTime, u.UpdateUserName, u.UpdateUserId, }).ExecuteCommandAsync(); } // 保存分页游标 if (userIdsRes.Result.NextCursor == null) break; offset = (int)userIdsRes.Result.NextCursor; } var sysUser = await _sysUserRep.AsQueryable() .Select(u => new { u.Id, u.Account, u.Phone }).ToListAsync(); var dingTalkUser = await _dingTalkUserRepo.AsQueryable() .Where(u => sysUser.Any(m => m.Account == u.JobNumber)) .Select(u => new { u.Id, u.JobNumber, u.Mobile }).ToListAsync(); // 更新钉钉用户中系统用户Id var uDingTalkUsers = dingTalkUser.Select(u => new DingTalkUser { Id = u.Id, SysUserId = sysUser.Where(m => m.Account == u.JobNumber).Select(m => m.Id).FirstOrDefault(), }).ToList(); if (uDingTalkUsers.Count > 0) { await _dingTalkUserRepo.AsUpdateable(uDingTalkUsers).UpdateColumns(u => new { u.SysUserId, u.UpdateTime, u.UpdateUserName, u.UpdateUserId, }).ExecuteCommandAsync(); } return; } /// /// 获取企业内部应用的access_token 🔖 /// /// /// [DisplayName("获取企业内部应用的access_token")] public async Task GetDingTalkToken([FromQuery] GetDingTalkTokenInput input) { return await _dingTalkApi.GetDingTalkToken(input); } /// /// 获取在职员工列表 🔖 /// /// /// /// [DisplayName("获取在职员工列表")] public async Task> GetDingTalkCurrentEmployeesList(string access_token, [Required] GetDingTalkCurrentEmployeesListInput input) { return await _dingTalkApi.GetDingTalkCurrentEmployeesList(access_token, input); } /// /// 获取员工花名册字段信息 🔖 /// /// /// /// [DisplayName("获取员工花名册字段信息")] public async Task>> GetDingTalkCurrentEmployeesRosterList(string access_token, [Required] GetDingTalkCurrentEmployeesRosterListInput input) { return await _dingTalkApi.GetDingTalkCurrentEmployeesRosterList(access_token, input); } /// /// 发送钉钉互动卡片 🔖 /// /// /// /// [DisplayName("给指定用户发送钉钉互动卡片")] public async Task DingTalkSendInteractiveCards(string token, DingTalkSendInteractiveCardsInput input) { return await _dingTalkApi.DingTalkSendInteractiveCards(token, input); } }