QiYeWeChatApiService.cs 68 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576
  1. using Microsoft.AspNetCore.Identity;
  2. using NPOI.Util;
  3. using OASystem.Domain.Dtos.QiYeWeChat;
  4. using OASystem.Domain.ViewModels.JuHeExchangeRate;
  5. using OASystem.Domain.ViewModels.QiYeWeChat;
  6. using Org.BouncyCastle.Ocsp;
  7. using SqlSugar;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Diagnostics;
  11. using System.Text.Json;
  12. using Ubiety.Dns.Core;
  13. using Ubiety.Dns.Core.Common;
  14. namespace OASystem.API.OAMethodLib.QiYeWeChatAPI
  15. {
  16. /// <summary>
  17. /// 聚合Api 服务
  18. /// </summary>
  19. public class QiYeWeChatApiService : IQiYeWeChatApiService
  20. {
  21. private readonly HttpClient _httpClient;
  22. private readonly string CorpId = "wwe978bef5495a0728"; //企业Id
  23. private readonly string PersonnelAssistant_AgentId = "3010011"; //人事助手Id
  24. private readonly string PersonnelAssistant_Corpsecret = "ig--IJd6TxWDMJ1wT4e-RDRcRX12v5GjB359DNATwJ4"; //人事助手凭证密钥
  25. private readonly string PunchCard_AgentId = "3010185"; //打卡Id
  26. private readonly string PunchCard_Corpsecret = "Xhrl37GOqlAjsu0VzUSJECaJdjzkDXQLbvrzRsZQb8M"; //打卡凭证密钥
  27. private readonly string Email_AgentId = "1000004"; //E-Mail Id
  28. private readonly string Email_Corpsecret = "NA1zbJM15GmgjPYwDOqz59dIo1Wnug-MbU107MeUemc"; //E-Mail 凭证密钥
  29. private readonly string AddressBook_Corpsecret = "Y1tnjh7j-BvbqAytAoXZPUbmDR6dqLTL6mXtc6PZ7fo"; //通讯录同步 凭证密钥
  30. private readonly string Approve_AgentId = "3010040"; //审批 Id
  31. private readonly string Approve_Corpsecret = "k_Jo69Jw9Hqg_in-Rypbs30PNbxOYa1t4e-dxYuT-kw"; //审批 凭证密钥
  32. private readonly string GroupStatus_AgentId = "1000008"; //OA通知 Id
  33. private readonly string GroupStatus_Corpsecret = "7J_ST3jTPzbZpFwl7ttToTVufjEx6O2wuApvKHxt2Ak"; //OA通知Secret
  34. private readonly DateTime _1970 = new DateTime(1970, 1, 1, 0, 0, 0, 0);
  35. private readonly JobPostRepository _jobPostRep;
  36. /// <summary>
  37. /// 初始化
  38. /// </summary>
  39. /// <param name="clientFactory"></param>
  40. /// <param name="jobPostRep"></param>
  41. public QiYeWeChatApiService(IHttpClientFactory clientFactory, JobPostRepository jobPostRep)
  42. {
  43. _httpClient = clientFactory.CreateClient("PublicQiYeWeChatApi"); ;
  44. _jobPostRep = jobPostRep;
  45. }
  46. /// <summary>
  47. /// 获取access_token
  48. /// </summary>
  49. /// <param name="applicationType">
  50. /// 1:人事助手
  51. /// 2:打卡
  52. /// 3:邮件
  53. /// 4:通讯录同步
  54. /// 5:审批
  55. /// 6:团组状态通知
  56. /// </param>
  57. /// <returns></returns>
  58. private async Task<Access_TokenView> GetTokenAsync(int applicationType)
  59. {
  60. Access_TokenView tokenResult = new Access_TokenView();
  61. Access_Token_Request access_Token = new Access_Token_Request();
  62. string cacheName = string.Empty;
  63. if (applicationType == 1)
  64. {
  65. access_Token.corpsecret = PersonnelAssistant_Corpsecret; //人事助手
  66. cacheName = "PersonnelAssistant_Access_Token";
  67. }
  68. else if (applicationType == 2)
  69. {
  70. access_Token.corpsecret = PunchCard_Corpsecret; //打卡
  71. cacheName = "PunchCard_Access_Token";
  72. }
  73. else if (applicationType == 3)
  74. {
  75. access_Token.corpsecret = Email_Corpsecret; //E-Mail
  76. cacheName = "Enail_Access_Token";
  77. }
  78. else if (applicationType == 4) //通讯录同步
  79. {
  80. access_Token.corpsecret = AddressBook_Corpsecret;
  81. cacheName = "AddressBook_Access_Token";
  82. }
  83. else if (applicationType == 5) //审批
  84. {
  85. access_Token.corpsecret = Approve_Corpsecret;
  86. cacheName = "Approve_Access_Token";
  87. }
  88. else if (applicationType == 6) //团组状态通知
  89. {
  90. access_Token.corpsecret = GroupStatus_Corpsecret;
  91. cacheName = "GroupStatus_Access_Token";
  92. }
  93. else
  94. {
  95. tokenResult.errmsg = "未识别应用类型Id!";
  96. return tokenResult;
  97. }
  98. string access_token = await RedisRepository.RedisFactory
  99. .CreateRedisRepository()
  100. .StringGetAsync<string>(cacheName);//string 取
  101. if (string.IsNullOrEmpty(access_token))
  102. {
  103. string access_token_url = string.Format(@"/cgi-bin/gettoken?corpid={0}&corpsecret={1}", access_Token.corpid, access_Token.corpsecret);
  104. var access_tokenReq = await _httpClient.GetAsync(access_token_url);
  105. if (access_tokenReq.IsSuccessStatusCode)
  106. {
  107. var stringResponse = await access_tokenReq.Content.ReadAsStringAsync();
  108. tokenResult = System.Text.Json.JsonSerializer.Deserialize<Access_TokenView>(stringResponse,
  109. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  110. if (tokenResult.errcode == 0)
  111. {
  112. TimeSpan ts = DateTime.Now.AddMinutes(118) - DateTime.Now; //设置redis 过期时间 118分钟
  113. await RedisRepository
  114. .RedisFactory
  115. .CreateRedisRepository()
  116. .StringSetAsync<string>(cacheName, tokenResult.access_token, ts);//string 存
  117. }
  118. }
  119. else
  120. {
  121. tokenResult.errmsg = "企业微信Token未获取到!";
  122. }
  123. }
  124. else
  125. {
  126. tokenResult.errcode = 0;
  127. tokenResult.access_token = access_token;
  128. }
  129. return tokenResult;
  130. }
  131. #region 添加员工
  132. /// <summary>
  133. /// 获取成员ID列表
  134. /// </summary>
  135. /// <returns></returns>
  136. public async Task<UserIdListView> GetUserIdListAsync()
  137. {
  138. UserIdListView userIdListView = new UserIdListView();
  139. Access_TokenView access_Token = await GetTokenAsync(4);
  140. if (access_Token.errcode != 0)
  141. {
  142. userIdListView.errcode = access_Token.errcode;
  143. userIdListView.errmsg = access_Token.errmsg;
  144. return userIdListView;
  145. }
  146. string url = string.Format("/cgi-bin/user/list_id?access_token={0}&debug=1", access_Token.access_token);
  147. var jsonData = new
  148. {
  149. access_token = access_Token.access_token, //调用接口凭证
  150. };
  151. var json = System.Text.Json.JsonSerializer.Serialize(jsonData);
  152. var content = new StringContent(json, Encoding.UTF8, "application/json");
  153. var create_Req = await _httpClient.PostAsync(url, content);
  154. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  155. userIdListView = System.Text.Json.JsonSerializer.Deserialize<UserIdListView>(stringResponse,
  156. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  157. return userIdListView;
  158. }
  159. /// <summary>
  160. /// 获取成员信息
  161. /// </summary>
  162. /// <returns></returns>
  163. public async Task<QYWX_UserInfosView> GetUserInfosAsync()
  164. {
  165. QYWX_UserInfosView infoViews = new QYWX_UserInfosView();
  166. //获取员工useridList
  167. UserIdListView useridView = await GetUserIdListAsync();
  168. if (useridView.errcode != 0)
  169. {
  170. infoViews.errcode = useridView.errcode;
  171. infoViews.errmsg = useridView.errmsg;
  172. return infoViews;
  173. }
  174. Access_TokenView access_Token = await GetTokenAsync(4);
  175. if (access_Token.errcode != 0)
  176. {
  177. infoViews.errcode = access_Token.errcode;
  178. infoViews.errmsg = access_Token.errmsg;
  179. return infoViews;
  180. }
  181. List<QYWX_UserInfo> items = new List<QYWX_UserInfo>();
  182. string urlTitle = string.Format("/cgi-bin/user/get?access_token={0}", access_Token.access_token);
  183. foreach (var item in useridView.dept_user)
  184. {
  185. if (!string.IsNullOrEmpty(item.userid))
  186. {
  187. QYWX_UserInfoView _UserInfoView = new QYWX_UserInfoView();
  188. string url = string.Format("{0}&userid={1}", urlTitle, item.userid);
  189. var create_Req = await _httpClient.GetAsync(url);
  190. if (create_Req.IsSuccessStatusCode)
  191. {
  192. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  193. _UserInfoView = System.Text.Json.JsonSerializer.Deserialize<QYWX_UserInfoView>(stringResponse,
  194. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  195. if (_UserInfoView.errcode == 0)
  196. {
  197. items.Add(new QYWX_UserInfo()
  198. {
  199. userid = _UserInfoView.QYWX_User.userid,
  200. name = _UserInfoView.QYWX_User.name,
  201. mobile = _UserInfoView.QYWX_User.mobile,
  202. position = _UserInfoView.QYWX_User.position,
  203. email = _UserInfoView.QYWX_User.email,
  204. biz_mail = _UserInfoView.QYWX_User.biz_mail,
  205. });
  206. }
  207. }
  208. }
  209. }
  210. infoViews.errcode = 0;
  211. infoViews.errmsg = "操作成功!";
  212. infoViews.QYWX_Users = items;
  213. return infoViews;
  214. }
  215. /// <summary>
  216. /// 获取部门列表
  217. /// </summary>
  218. /// <param name="access_token"></param>
  219. /// <returns></returns>
  220. private async Task<DepartmentListView> GetDepartmentAsync(string access_token)
  221. {
  222. DepartmentListView result = new DepartmentListView();
  223. string dep_url = string.Format("/cgi-bin/department/list?access_token={0}&debug=1", access_token);
  224. var depReq = await _httpClient.GetAsync(dep_url);
  225. if (depReq.IsSuccessStatusCode)
  226. {
  227. var stringResponse = await depReq.Content.ReadAsStringAsync();
  228. result = System.Text.Json.JsonSerializer.Deserialize<DepartmentListView>(stringResponse,
  229. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  230. }
  231. else
  232. {
  233. result.errcode = -1;
  234. result.errmsg = "企业微信部门列表未获取到!";
  235. }
  236. return result;
  237. }
  238. /// <summary>
  239. /// 创建员工
  240. /// </summary>
  241. /// <param name="create_Request"></param>
  242. /// <returns></returns>
  243. public async Task<ResponseBase> CreateAsync(Create_Request create_Request)
  244. {
  245. ResponseBase responseBase = new ResponseBase();
  246. #region 职位
  247. string depName = string.Empty,
  248. jobName = string.Empty;
  249. List<Sys_JobPostI> jobs = await _jobPostRep.QueryJobPost(string.Empty);
  250. Sys_JobPostI jobPost = jobs.Where(it => it.IsDel == 0 && it.Id == Convert.ToInt32(create_Request.position)).FirstOrDefault();
  251. if (jobPost != null)
  252. {
  253. depName = jobPost.DepName;
  254. jobName = jobPost.JobName;
  255. }
  256. create_Request.position = jobName;
  257. if (string.IsNullOrEmpty(create_Request.biz_mail))
  258. {
  259. create_Request.biz_mail = string.Format("{0}@pan-american-intl.com", create_Request.userid);
  260. }
  261. #endregion
  262. Access_TokenView access_Token = new Access_TokenView();
  263. access_Token = await GetTokenAsync(4);
  264. if (access_Token.errcode == 0)
  265. {
  266. create_Request.access_token = access_Token.access_token;
  267. #region 处理部门
  268. Access_TokenView dep_Access_Token = await GetTokenAsync(1);
  269. if (dep_Access_Token.errcode != 0)
  270. {
  271. responseBase.errcode = dep_Access_Token.errcode;
  272. responseBase.errmsg = dep_Access_Token.errmsg;
  273. }
  274. DepartmentListView depData = await GetDepartmentAsync(dep_Access_Token.access_token);
  275. if (depData.errcode == 0)
  276. {
  277. var depData1 = depData.department.Where(x => x.name == depName).FirstOrDefault();
  278. if (depData1 != null)
  279. {
  280. create_Request.department = new List<long> { depData1.id };
  281. }
  282. else
  283. {
  284. responseBase.errcode = -1;
  285. responseBase.errmsg = "未在企业微信上找到OA内设置的部门";
  286. return responseBase;
  287. }
  288. }
  289. else
  290. {
  291. responseBase.errcode = depData.errcode;
  292. responseBase.errmsg = depData.errmsg;
  293. return responseBase;
  294. }
  295. #endregion
  296. create_Request.access_token = access_Token.access_token;
  297. string create_url = string.Format("/cgi-bin/user/create?access_token={0}", create_Request.access_token);
  298. var json = System.Text.Json.JsonSerializer.Serialize(create_Request);
  299. var content = new StringContent(json, Encoding.UTF8, "application/json");
  300. var create_Req = await _httpClient.PostAsync(create_url, content);
  301. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  302. responseBase = System.Text.Json.JsonSerializer.Deserialize<ResponseBase>(stringResponse,
  303. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  304. }
  305. else
  306. {
  307. responseBase.errcode = access_Token.errcode;
  308. responseBase.errmsg = access_Token.errmsg;
  309. }
  310. return responseBase;
  311. }
  312. #endregion
  313. #region 打卡
  314. /// <summary>
  315. /// 获取企业所有打卡规则
  316. /// </summary>
  317. /// <returns></returns>
  318. public async Task<CorpCheckInRuleView> GetCheckIn_CorpCheckInOptionAsync()
  319. {
  320. CorpCheckInRuleView corpCheckInRuleView = new CorpCheckInRuleView();
  321. //获取打卡数据 token
  322. Access_TokenView access_Token = await GetTokenAsync(2);
  323. if (access_Token.errcode != 0)
  324. {
  325. corpCheckInRuleView.errcode = access_Token.errcode;
  326. corpCheckInRuleView.errmsg = string.Format("【企业微信】【获取月打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
  327. return corpCheckInRuleView;
  328. }
  329. string url = string.Format("/cgi-bin/checkin/getcorpcheckinoption?access_token={0}", access_Token.access_token);
  330. var content = new StringContent(string.Empty, Encoding.UTF8, "application/json");
  331. var create_Req = await _httpClient.PostAsync(url, content);
  332. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  333. corpCheckInRuleView = System.Text.Json.JsonSerializer.Deserialize<CorpCheckInRuleView>(stringResponse,
  334. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  335. return corpCheckInRuleView;
  336. }
  337. /// <summary>
  338. /// 获取月打卡记录数据
  339. /// </summary>
  340. /// <param name="startDt"></param>
  341. /// <param name="endDt"></param>
  342. /// <returns></returns>
  343. public async Task<CheckInView> GetCheckin_MonthDataAsync1(DateTime startDt, DateTime endDt)
  344. {
  345. CheckInView checkInView = new CheckInView();
  346. //获取员工Id
  347. UserIdListView userIdListView = await GetUserIdListAsync();
  348. if (userIdListView.errcode != 0)
  349. {
  350. checkInView.errcode = userIdListView.errcode;
  351. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】{0}", userIdListView.errmsg);
  352. return checkInView;
  353. }
  354. if (userIdListView.dept_user == null || userIdListView.dept_user.Count <= 0)
  355. {
  356. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】未查出员工Id");
  357. return checkInView;
  358. }
  359. //获取打卡数据 token
  360. Access_TokenView access_Token = await GetTokenAsync(2);
  361. if (access_Token.errcode != 0)
  362. {
  363. checkInView.errcode = access_Token.errcode;
  364. checkInView.errmsg = string.Format("【企业微信】【获取月打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
  365. return checkInView;
  366. }
  367. string url = string.Format("/cgi-bin/checkin/getcheckin_monthdata?access_token={0}", access_Token.access_token);
  368. Checkin_MonthData_Request checkInReq = new Checkin_MonthData_Request();
  369. checkInReq.access_token = access_Token.access_token;
  370. checkInReq.starttime = (uint)(startDt - _1970).TotalSeconds;
  371. checkInReq.endtime = (uint)(endDt - _1970).TotalSeconds;
  372. checkInReq.useridlist = userIdListView.dept_user.Select(it => it.userid).ToList();
  373. var json = System.Text.Json.JsonSerializer.Serialize(checkInReq);
  374. var content = new StringContent(json, Encoding.UTF8, "application/json");
  375. var create_Req = await _httpClient.PostAsync(url, content);
  376. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  377. checkInView = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponse,
  378. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  379. return checkInView;
  380. }
  381. /// <summary>
  382. /// 获取月打卡记录数据
  383. /// </summary>
  384. /// <param name="startDt"></param>
  385. /// <param name="endDt"></param>
  386. /// <returns></returns>
  387. public async Task<CheckInView> GetCheckin_MonthDataAsync(DateTime startDt, DateTime endDt)
  388. {
  389. CheckInView checkInView = new CheckInView();
  390. //获取员工Id
  391. UserIdListView userIdListView = await GetUserIdListAsync();
  392. if (userIdListView.errcode != 0)
  393. {
  394. checkInView.errcode = userIdListView.errcode;
  395. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】{0}", userIdListView.errmsg);
  396. return checkInView;
  397. }
  398. if (userIdListView.dept_user == null || userIdListView.dept_user.Count <= 0)
  399. {
  400. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】未查出员工Id");
  401. return checkInView;
  402. }
  403. //获取打卡数据 token
  404. Access_TokenView access_Token = await GetTokenAsync(2);
  405. if (access_Token.errcode != 0)
  406. {
  407. checkInView.errcode = access_Token.errcode;
  408. checkInView.errmsg = string.Format("【企业微信】【获取月打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
  409. return checkInView;
  410. }
  411. #region 处理超31天的月报表数据
  412. string url = string.Format("/cgi-bin/checkin/getcheckin_monthdata?access_token={0}", access_Token.access_token);
  413. Checkin_MonthData_Request checkInReq = new Checkin_MonthData_Request();
  414. checkInReq.access_token = access_Token.access_token;
  415. checkInReq.useridlist = userIdListView.dept_user.Select(it => it.userid).ToList();
  416. uint starttimeOne, endtimeOne, starttimeTwo, endtimeTwo;
  417. int days = (int)(endDt - startDt).TotalDays;
  418. if (days > 32)
  419. {
  420. starttimeOne = (uint)((startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  421. endtimeOne = (uint)((startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  422. starttimeTwo = (uint)((startDt.AddDays(32).ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  423. endtimeTwo = (uint)((endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  424. checkInReq.starttime = starttimeOne;
  425. checkInReq.endtime = endtimeOne;
  426. CheckInView checkInViewOne = new CheckInView();
  427. var jsonOne = System.Text.Json.JsonSerializer.Serialize(checkInReq);
  428. var contentOne = new StringContent(jsonOne, Encoding.UTF8, "application/json");
  429. var create_ReqOne = await _httpClient.PostAsync(url, contentOne);
  430. var stringResponseOne = await create_ReqOne.Content.ReadAsStringAsync();
  431. checkInViewOne = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponseOne,
  432. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  433. if (checkInViewOne.errcode != 0)
  434. {
  435. return checkInViewOne;
  436. }
  437. checkInReq.starttime = starttimeTwo;
  438. checkInReq.endtime = endtimeTwo;
  439. CheckInView checkInViewTwo = new CheckInView();
  440. var jsonTwo = System.Text.Json.JsonSerializer.Serialize(checkInReq);
  441. var contentTwo = new StringContent(jsonTwo, Encoding.UTF8, "application/json");
  442. var create_ReqTwo = await _httpClient.PostAsync(url, contentTwo);
  443. var stringResponseTwo = await create_ReqTwo.Content.ReadAsStringAsync();
  444. checkInViewTwo = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponseTwo,
  445. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  446. if (checkInViewTwo.errcode != 0)
  447. {
  448. return checkInViewTwo;
  449. }
  450. #region 整合月报表数据
  451. foreach (var item_one in checkInViewOne.datas)
  452. {
  453. foreach (var item_two in checkInViewTwo.datas)
  454. {
  455. if (item_one.base_info.acctid.Equals(item_two.base_info.acctid))
  456. {
  457. #region summary_info 汇总信息整合
  458. item_one.summary_info.work_days += item_two.summary_info.work_days; //应打卡天数
  459. item_one.summary_info.regular_days += item_two.summary_info.regular_days; //正常天数
  460. item_one.summary_info.except_days += item_two.summary_info.except_days; //异常天数
  461. item_one.summary_info.regular_work_sec += item_two.summary_info.regular_work_sec; //实际工作时长,为统计周期每日实际工作时长之和
  462. item_one.summary_info.standard_work_sec += item_two.summary_info.standard_work_sec; //标准工作时长,为统计周期每日标准工作时长之和
  463. #endregion
  464. #region exception_infos 异常状态统计信息 整合
  465. item_one.exception_infos.AddRange(item_two.exception_infos);
  466. #endregion
  467. #region sp_items 假勤统计信息整合
  468. item_one.sp_items.AddRange(item_two.sp_items);
  469. #endregion
  470. #region overwork_info 加班情况 整合
  471. item_one.overwork_info.workday_over_sec += item_two.overwork_info.workday_over_sec; //工作日加班时长
  472. item_one.overwork_info.holidays_over_sec += item_two.overwork_info.holidays_over_sec; //节假日加班时长
  473. item_one.overwork_info.restdays_over_sec += item_two.overwork_info.restdays_over_sec; //休息日加班时长
  474. #endregion
  475. break;
  476. }
  477. }
  478. }
  479. #endregion
  480. checkInView = checkInViewOne;
  481. }
  482. else
  483. {
  484. checkInReq.starttime = (uint)((startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  485. checkInReq.endtime = (uint)((endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000);
  486. var json = System.Text.Json.JsonSerializer.Serialize(checkInReq);
  487. var content = new StringContent(json, Encoding.UTF8, "application/json");
  488. var create_Req = await _httpClient.PostAsync(url, content);
  489. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  490. checkInView = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponse,
  491. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  492. }
  493. #endregion
  494. return checkInView;
  495. }
  496. /// <summary>
  497. /// 获取月打卡数据(redis缓存)
  498. /// </summary>
  499. /// <param name="startDt"></param>
  500. /// <param name="endDt"></param>
  501. /// <returns></returns>
  502. public async Task<CheckInView> GetCheckin_MonthDataRedisAsync(DateTime startDt, DateTime endDt)
  503. {
  504. CheckInView checkInView = new CheckInView();
  505. string checkInDatastring = await RedisRepository.RedisFactory
  506. .CreateRedisRepository()
  507. .StringGetAsync<string>("Checkin_MonthData");//List<T> 取
  508. if (!string.IsNullOrEmpty(checkInDatastring))
  509. {
  510. checkInView = JsonConvert.DeserializeObject<CheckInView>(checkInDatastring);
  511. return checkInView;
  512. }
  513. //获取员工Id
  514. UserIdListView userIdListView = await GetUserIdListAsync();
  515. if (userIdListView.errcode != 0)
  516. {
  517. checkInView.errcode = userIdListView.errcode;
  518. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】{0}", userIdListView.errmsg);
  519. return checkInView;
  520. }
  521. if (userIdListView.dept_user == null || userIdListView.dept_user.Count <= 0)
  522. {
  523. checkInView.errmsg = string.Format("【企业微信】【获取员工IdList】【Msg】未查出员工Id");
  524. return checkInView;
  525. }
  526. //获取月打卡数据 token
  527. Access_TokenView access_Token = await GetTokenAsync(2);
  528. if (access_Token.errcode != 0)
  529. {
  530. checkInView.errcode = access_Token.errcode;
  531. checkInView.errmsg = string.Format("【企业微信】【获取月打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
  532. return checkInView;
  533. }
  534. string url = string.Format("/cgi-bin/checkin/getcheckin_monthdata?access_token={0}", access_Token.access_token);
  535. Checkin_MonthData_Request checkInReq = new Checkin_MonthData_Request();
  536. checkInReq.access_token = access_Token.access_token;
  537. checkInReq.starttime = (uint)(startDt - _1970).TotalSeconds;
  538. checkInReq.endtime = (uint)(endDt - _1970).TotalSeconds;
  539. checkInReq.useridlist = userIdListView.dept_user.Select(it => it.userid).ToList();
  540. var json = System.Text.Json.JsonSerializer.Serialize(checkInReq);
  541. var content = new StringContent(json, Encoding.UTF8, "application/json");
  542. var create_Req = await _httpClient.PostAsync(url, content);
  543. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  544. checkInView = System.Text.Json.JsonSerializer.Deserialize<CheckInView>(stringResponse,
  545. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  546. if (checkInView.errcode == 0)
  547. {
  548. TimeSpan ts = DateTime.Now.AddMinutes(60) - DateTime.Now; //设置redis 过期时间 60分钟
  549. await RedisRepository
  550. .RedisFactory
  551. .CreateRedisRepository()
  552. .StringSetAsync<string>("Checkin_MonthData", JsonConvert.SerializeObject(checkInView), ts);//List<T> 存
  553. }
  554. return checkInView;
  555. }
  556. /// <summary>
  557. /// 获取打卡记录数据
  558. /// </summary>
  559. /// <param name="useridlist">需要获取打卡记录的用户列表</param>
  560. /// <param name="opencheckindatatype">打卡类型。1:上下班打卡;2:外出打卡;3:全部打卡</param>
  561. /// <param name="startDt">获取打卡记录的开始时间。Unix时间戳</param>
  562. /// <param name="endDt">获取打卡记录的结束时间。Unix时间戳</param>
  563. /// <returns></returns>
  564. public async Task<CheckInDataView> GetCheckinDataAsync(List<string> useridlist, int opencheckindatatype, DateTime startDt, DateTime endDt)
  565. {
  566. CheckInDataView checkInDataView = new CheckInDataView();
  567. //参数处理
  568. if (useridlist.Count <= 0)
  569. {
  570. checkInDataView.errmsg = "请填写企业微信用户id,可为多个!";
  571. return checkInDataView;
  572. }
  573. if (useridlist.Count > 100)
  574. {
  575. checkInDataView.errmsg = "用户列表不超过100个。若用户超过100个,请分批获取!";
  576. return checkInDataView;
  577. }
  578. //获取打卡数据 token
  579. Access_TokenView access_Token = await GetTokenAsync(2);
  580. if (access_Token.errcode != 0)
  581. {
  582. checkInDataView.errcode = access_Token.errcode;
  583. checkInDataView.errmsg = string.Format("【企业微信】【获取打卡数据】【Token】【Msg】{0}", access_Token.errmsg);
  584. return checkInDataView;
  585. }
  586. string url = string.Format("/cgi-bin/checkin/getcheckindata?access_token={0}", access_Token.access_token);
  587. DateTime centerDt = startDt.AddDays(30);
  588. long startTs = (long)(startDt - _1970).TotalSeconds;
  589. long centerTs = (long)(centerDt - _1970).TotalSeconds;
  590. long endTs = (long)(endDt - _1970).TotalSeconds;
  591. CheckInData_Request checkInData_Req = new CheckInData_Request()
  592. {
  593. access_token = access_Token.access_token,
  594. opencheckindatatype = opencheckindatatype,
  595. useridlist = useridlist,
  596. starttime = startTs,
  597. endtime = centerTs
  598. };
  599. var json = System.Text.Json.JsonSerializer.Serialize(checkInData_Req);
  600. var content = new StringContent(json, Encoding.UTF8, "application/json");
  601. var create_Req = await _httpClient.PostAsync(url, content);
  602. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  603. checkInDataView = System.Text.Json.JsonSerializer.Deserialize<CheckInDataView>(stringResponse,
  604. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  605. if (centerDt < endDt)
  606. {
  607. checkInData_Req.starttime = centerTs;
  608. checkInData_Req.endtime = endTs;
  609. var json1 = System.Text.Json.JsonSerializer.Serialize(checkInData_Req);
  610. var content1 = new StringContent(json1, Encoding.UTF8, "application/json");
  611. var create_Req1 = await _httpClient.PostAsync(url, content1);
  612. var stringResponse1 = await create_Req1.Content.ReadAsStringAsync();
  613. CheckInDataView checkInDataView1 = System.Text.Json.JsonSerializer.Deserialize<CheckInDataView>(stringResponse1,
  614. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  615. if (checkInDataView1.errcode != 0)
  616. {
  617. return checkInDataView1;
  618. }
  619. checkInDataView.checkindata.AddRange(checkInDataView1.checkindata);
  620. return checkInDataView;
  621. }
  622. return checkInDataView;
  623. }
  624. /// <summary>
  625. /// 获取打卡日报数据
  626. /// </summary>
  627. /// <param name="useridlist"></param>
  628. /// <param name="startDt"></param>
  629. /// <param name="endDt"></param>
  630. /// <returns></returns>
  631. public async Task<CheckInDayDataView> GetCheckInDayDataAsync1(List<string> useridlist, DateTime startDt, DateTime endDt)
  632. {
  633. CheckInDayDataView checkInDayDataView = new CheckInDayDataView();
  634. //参数处理
  635. if (useridlist.Count <= 0)
  636. {
  637. checkInDayDataView.errmsg = "请填写企业微信用户id,可为多个!";
  638. return checkInDayDataView;
  639. }
  640. //获取打卡数据 token
  641. Access_TokenView access_Token = await GetTokenAsync(2);
  642. if (access_Token.errcode != 0)
  643. {
  644. checkInDayDataView.errcode = access_Token.errcode;
  645. checkInDayDataView.errmsg = string.Format("【企业微信】【获取打卡日报数据】【Token】【Msg】{0}", access_Token.errmsg);
  646. return checkInDayDataView;
  647. }
  648. string url = string.Format("/cgi-bin/checkin/getcheckin_daydata?access_token={0}&debug=1", access_Token.access_token);
  649. long startTs = (startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  650. long endTs = (endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  651. var checkInData_Req = new
  652. {
  653. access_token = access_Token.access_token,
  654. useridlist = useridlist,
  655. starttime = startTs,
  656. endtime = endTs
  657. };
  658. var json = System.Text.Json.JsonSerializer.Serialize(checkInData_Req);
  659. var content = new StringContent(json, Encoding.UTF8, "application/json");
  660. var create_Req = await _httpClient.PostAsync(url, content);
  661. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  662. checkInDayDataView = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponse,
  663. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  664. return checkInDayDataView;
  665. }
  666. /// <summary>
  667. /// 获取打卡日报数据
  668. /// </summary>
  669. /// <param name="useridlist"></param>
  670. /// <param name="startDt"></param>
  671. /// <param name="endDt"></param>
  672. /// <returns></returns>
  673. public async Task<CheckInDayDataView> GetCheckInDayDataAsync(List<string> useridlist, DateTime startDt, DateTime endDt)
  674. {
  675. CheckInDayDataView checkInDayDataView = new CheckInDayDataView();
  676. //参数处理
  677. if (useridlist.Count <= 0)
  678. {
  679. checkInDayDataView.errmsg = "请填写企业微信用户id,可为多个!";
  680. return checkInDayDataView;
  681. }
  682. //获取打卡数据 token
  683. Access_TokenView access_Token = await GetTokenAsync(2);
  684. if (access_Token.errcode != 0)
  685. {
  686. checkInDayDataView.errcode = access_Token.errcode;
  687. checkInDayDataView.errmsg = string.Format("【企业微信】【获取打卡日报数据】【Token】【Msg】{0}", access_Token.errmsg);
  688. return checkInDayDataView;
  689. }
  690. string url = string.Format("/cgi-bin/checkin/getcheckin_daydata?access_token={0}&debug=1", access_Token.access_token);
  691. int days = (int)(endDt - startDt).TotalDays;
  692. if (days > 31)
  693. {
  694. long startTs_One = (startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  695. long endTs_One = (startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  696. long startTs_Two = (startDt.AddDays(31).ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  697. long endTs_Two = (endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  698. var checkInData_ReqOne = new
  699. {
  700. access_token = access_Token.access_token,
  701. useridlist = useridlist,
  702. starttime = startTs_One,
  703. endtime = endTs_One
  704. };
  705. var jsonOne = System.Text.Json.JsonSerializer.Serialize(checkInData_ReqOne);
  706. var contentOne = new StringContent(jsonOne, Encoding.UTF8, "application/json");
  707. var create_ReqOne = await _httpClient.PostAsync(url, contentOne);
  708. var stringResponseOne = await create_ReqOne.Content.ReadAsStringAsync();
  709. CheckInDayDataView checkInDayDataViewOne = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponseOne,
  710. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  711. if (checkInDayDataViewOne.errcode != 0)
  712. {
  713. return checkInDayDataViewOne;
  714. }
  715. var checkInData_ReqTwo = new
  716. {
  717. access_token = access_Token.access_token,
  718. useridlist = useridlist,
  719. starttime = startTs_Two,
  720. endtime = endTs_Two
  721. };
  722. var jsonTwo = System.Text.Json.JsonSerializer.Serialize(checkInData_ReqTwo);
  723. var contentTwo = new StringContent(jsonTwo, Encoding.UTF8, "application/json");
  724. var create_ReqTwo = await _httpClient.PostAsync(url, contentTwo);
  725. var stringResponseTwo = await create_ReqTwo.Content.ReadAsStringAsync();
  726. CheckInDayDataView checkInDayDataViewTwo = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponseTwo,
  727. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  728. if (checkInDayDataViewTwo.errcode != 0)
  729. {
  730. return checkInDayDataViewTwo;
  731. }
  732. checkInDayDataViewOne.datas.AddRange(checkInDayDataViewTwo.datas);
  733. return checkInDayDataViewOne;
  734. }
  735. else
  736. {
  737. long startTs = (startDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  738. long endTs = (endDt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  739. var checkInData_Req = new
  740. {
  741. access_token = access_Token.access_token,
  742. useridlist = useridlist,
  743. starttime = startTs,
  744. endtime = endTs
  745. };
  746. var json = System.Text.Json.JsonSerializer.Serialize(checkInData_Req);
  747. var content = new StringContent(json, Encoding.UTF8, "application/json");
  748. var create_Req = await _httpClient.PostAsync(url, content);
  749. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  750. checkInDayDataView = System.Text.Json.JsonSerializer.Deserialize<CheckInDayDataView>(stringResponse,
  751. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  752. }
  753. return checkInDayDataView;
  754. }
  755. #endregion
  756. #region 审批
  757. /// <summary>
  758. /// 获取审批数据(旧)
  759. /// </summary>
  760. /// <param name="startDt"></param>
  761. /// <param name="endDt"></param>
  762. /// <returns></returns>
  763. public async Task<ApprovalDataView> GetApprovalDataAsync(DateTime startDt, DateTime endDt)
  764. {
  765. ApprovalDataView approvalDataView = new ApprovalDataView();
  766. //获取审批数据 token
  767. Access_TokenView access_Token = await GetTokenAsync(5);
  768. if (access_Token.errcode != 0)
  769. {
  770. approvalDataView.errcode = access_Token.errcode;
  771. approvalDataView.errmsg = string.Format("【企业微信】【获取审批数据】【Token】【Msg】{0}", access_Token.errmsg);
  772. return approvalDataView;
  773. }
  774. string url = string.Format("/cgi-bin/corp/getapprovaldata?access_token={0}", access_Token.access_token);
  775. ApprovalData_Request approvalDataReq = new ApprovalData_Request();
  776. approvalDataReq.access_token = access_Token.access_token;
  777. approvalDataReq.starttime = (uint)(startDt - _1970).TotalSeconds;
  778. approvalDataReq.endtime = (uint)(endDt - _1970).TotalSeconds;
  779. var json = System.Text.Json.JsonSerializer.Serialize(approvalDataReq);
  780. var content = new StringContent(json, Encoding.UTF8, "application/json");
  781. var create_Req = await _httpClient.PostAsync(url, content);
  782. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  783. approvalDataView = System.Text.Json.JsonSerializer.Deserialize<ApprovalDataView>(stringResponse,
  784. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  785. List<Sp_Info> sp_datas = new List<Sp_Info>();
  786. sp_datas.AddRange(approvalDataView.data);
  787. int index = 0;
  788. //多次访问审批接口
  789. if (approvalDataView.total >= 100)
  790. {
  791. approvalDataView.total -= 100;
  792. int forTotal = approvalDataView.total % 100 == 0 ? approvalDataView.total / 100 : approvalDataView.total / 100 + 1;
  793. long? next_spnum = approvalDataView.next_spnum;
  794. approvalDataReq.next_spnum = next_spnum;
  795. for (int i = 0; i < forTotal; i++)
  796. {
  797. index++;
  798. var for_json = System.Text.Json.JsonSerializer.Serialize(approvalDataReq);
  799. var for_content = new StringContent(for_json, Encoding.UTF8, "application/json");
  800. var for_create_Req = await _httpClient.PostAsync(url, for_content);
  801. var for_stringResponse = await for_create_Req.Content.ReadAsStringAsync();
  802. ApprovalDataView for_approvalDataView = System.Text.Json.JsonSerializer.Deserialize<ApprovalDataView>(for_stringResponse,
  803. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  804. approvalDataReq.next_spnum = for_approvalDataView.next_spnum; //重新定义游标
  805. sp_datas.AddRange(for_approvalDataView.data); //追加数据
  806. }
  807. approvalDataView.total += 100;
  808. }
  809. approvalDataView.data = sp_datas;
  810. return approvalDataView;
  811. }
  812. /// <summary>
  813. /// 获取审批数据(旧)(redis缓存)
  814. /// </summary>
  815. /// <param name="startDt"></param>
  816. /// <param name="endDt"></param>
  817. /// <returns></returns>
  818. public async Task<List<Sp_Info>> GetApprovalDatasAsync(DateTime startDt, DateTime endDt)
  819. {
  820. List<Sp_Info> sp_Infos = new List<Sp_Info>();
  821. //获取所有打卡补卡,审批 数据 前后范围增加10天
  822. DateTime sp_startDt = startDt.AddDays(-10);
  823. DateTime sp_centerDt = sp_startDt.AddDays(30);
  824. DateTime sp_endDt = endDt.AddDays(10);
  825. ApprovalDataView approvalData_1 = await GetApprovalDataAsync(sp_startDt, sp_centerDt); //时间段内所有 审批数据
  826. ApprovalDataView approvalData_2 = await GetApprovalDataAsync(sp_centerDt, sp_endDt); //时间段内所有 审批数据
  827. if (approvalData_1.errcode != 0)
  828. {
  829. Log.Error("企业微信 获取 " + sp_startDt + " - " + sp_centerDt + " 内审批 Msg:" + approvalData_1.errmsg);
  830. return sp_Infos;
  831. }
  832. sp_Infos.AddRange(approvalData_1.data);
  833. if (approvalData_2.errcode != 0)
  834. {
  835. Log.Error("企业微信 获取 " + sp_centerDt + " - " + sp_endDt + " 内审批 Msg:" + approvalData_2.errmsg);
  836. return sp_Infos;
  837. }
  838. sp_Infos.AddRange(approvalData_2.data);
  839. sp_Infos = sp_Infos.Where(it => it.sp_status == 2).ToList(); //存储已审核的数据
  840. return sp_Infos;
  841. }
  842. /// <summary>
  843. /// 获取审批数据(旧)(redis缓存)
  844. /// </summary>
  845. /// <param name="startDt"></param>
  846. /// <param name="endDt"></param>
  847. /// <returns></returns>
  848. public async Task<List<Sp_Info>> GetApprovalDatasRedisAsync(DateTime startDt, DateTime endDt)
  849. {
  850. List<Sp_Info> sp_Infos = new List<Sp_Info>();
  851. //获取所有打卡补卡,审批 数据 前后范围增加10天
  852. DateTime sp_startDt = startDt.AddDays(-10);
  853. DateTime sp_centerDt = sp_startDt.AddDays(30);
  854. DateTime sp_endDt = endDt.AddDays(10);
  855. string redisName = "ApprovalData" + sp_startDt.ToString("yyyyMMdd") + "-" + sp_endDt.ToString("yyyyMMdd");
  856. string sp_InfosString = string.Empty;
  857. //sp_InfosString = await RedisRepository.RedisFactory
  858. // .CreateRedisRepository()
  859. // .StringGetAsync<string>(redisName);//string 取
  860. if (string.IsNullOrEmpty(sp_InfosString))
  861. {
  862. ApprovalDataView approvalData_1 = await GetApprovalDataAsync(sp_startDt, sp_centerDt); //时间段内所有 审批数据
  863. ApprovalDataView approvalData_2 = await GetApprovalDataAsync(sp_centerDt, sp_endDt); //时间段内所有 审批数据
  864. if (approvalData_1.errcode != 0)
  865. {
  866. Log.Error("企业微信 获取 " + sp_startDt + " - " + sp_centerDt + " 内审批 Msg:" + approvalData_1.errmsg);
  867. return sp_Infos;
  868. }
  869. sp_Infos.AddRange(approvalData_1.data);
  870. if (approvalData_2.errcode != 0)
  871. {
  872. Log.Error("企业微信 获取 " + sp_centerDt + " - " + sp_endDt + " 内审批 Msg:" + approvalData_2.errmsg);
  873. return sp_Infos;
  874. }
  875. sp_Infos.AddRange(approvalData_2.data);
  876. sp_Infos = sp_Infos.Where(it => it.sp_status == 2).ToList(); //存储已审核的数据
  877. TimeSpan ts = DateTime.Now.AddMinutes(60) - DateTime.Now; //设置redis 过期时间 60分钟
  878. await RedisRepository
  879. .RedisFactory
  880. .CreateRedisRepository()
  881. .StringSetAsync<string>(redisName, JsonConvert.SerializeObject(sp_Infos), ts);//string 存
  882. }
  883. else
  884. {
  885. sp_Infos = JsonConvert.DeserializeObject<List<Sp_Info>>(sp_InfosString);
  886. }
  887. return sp_Infos;
  888. }
  889. /// <summary>
  890. /// 批量获取审批单号
  891. /// </summary>
  892. /// <param name="startDt"></param>
  893. /// <param name="endDt"></param>
  894. /// <param name="creator">
  895. /// 申请人
  896. /// </param>
  897. /// <param name="sp_status">
  898. /// sp_status-审批单状态(1-审批中;2-已通过;3-已驳回;4-已撤销;6-通过后撤销;7-已删除;10-已支付)
  899. /// </param>
  900. /// <param name="record_type">
  901. /// record_type == 0 ALL 有值 请填写对应值
  902. /// record_type - 审批单类型属性,1-请假;2-打卡补卡;3-出差;4-外出;5-加班; 6- 调班;7-会议室预定;8-退款审批;9-红包报销审批
  903. /// </param>
  904. /// <returns></returns>
  905. private async Task<ApprovalInfoView> GetApprovalInfoAsync(DateTime startDt, DateTime endDt, string creator, int sp_status, int record_type)
  906. {
  907. Stopwatch sw = new Stopwatch();
  908. sw.Start();
  909. ApprovalInfoView approvalInfoView = new ApprovalInfoView();
  910. //获取所有打卡补卡,审批 数据 向前增加30天 向后范围增加15天
  911. DateTime sp_startDt = startDt.AddDays(-30);
  912. DateTime sp_endDt = endDt.AddDays(15);
  913. //获取审批数据 token
  914. Access_TokenView access_Token = await GetTokenAsync(5);
  915. if (access_Token.errcode != 0)
  916. {
  917. approvalInfoView.errcode = access_Token.errcode;
  918. approvalInfoView.errmsg = string.Format("【企业微信】【批量获取审批单号】【Token】【Msg】{0}", access_Token.errmsg);
  919. return approvalInfoView;
  920. }
  921. string url = string.Format("/cgi-bin/oa/getapprovalinfo?access_token={0}", access_Token.access_token);
  922. List<Dic> filters = new List<Dic> {
  923. new Dic() { key = "creator", value = creator }, //筛选条件 申请人
  924. new Dic() { key = "sp_status", value = sp_status.ToString() }, //筛选条件 审核状态 2 已同意
  925. };
  926. if (record_type != 0)
  927. {
  928. filters.Add(new Dic() { key = "record_type", value = record_type.ToString() }); //筛选条件 审批单类型 请假 1
  929. }
  930. int days = (sp_endDt - sp_startDt).Days;
  931. decimal runCount = days % 30;
  932. int runCount1 = 0;
  933. if (runCount == 0) runCount1 = (int)(days % 30.00M);
  934. else runCount1 = (int)Math.Ceiling(days / 30.00M);
  935. List<string> sp_no_lists = new List<string>();
  936. for (int i = 0; i < runCount1; i++)
  937. {
  938. DateTime for_sp_startDt = sp_startDt;
  939. DateTime for_sp_endDt = sp_startDt.AddDays(30);
  940. sp_startDt = sp_startDt.AddDays(30);
  941. if (for_sp_endDt >= sp_endDt)
  942. {
  943. for_sp_endDt = sp_endDt;
  944. }
  945. ApprovalInfo_Request approvalInfoReq = new ApprovalInfo_Request()
  946. {
  947. access_token = access_Token.access_token,
  948. starttime = (uint)ConvertToTimeSpan(for_sp_startDt),
  949. endtime = (uint)ConvertToTimeSpan(for_sp_endDt),
  950. filters = filters.ToList(),
  951. };
  952. var json = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
  953. var content = new StringContent(json, Encoding.UTF8, "application/json");
  954. var create_Req = await _httpClient.PostAsync(url, content);
  955. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  956. ApprovalInfoView approvalInfoView1 = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(stringResponse,
  957. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  958. #region 分页调用 new_next_cursor != null
  959. string new_cursor = approvalInfoView1.new_next_cursor;
  960. while (!string.IsNullOrEmpty(new_cursor))
  961. {
  962. approvalInfoReq.new_cursor = new_cursor;
  963. var json1 = System.Text.Json.JsonSerializer.Serialize(approvalInfoReq);
  964. var content1 = new StringContent(json, Encoding.UTF8, "application/json");
  965. var create_Req1 = await _httpClient.PostAsync(url, content1);
  966. var stringResponse1 = await create_Req1.Content.ReadAsStringAsync();
  967. ApprovalInfoView approvalInfoView11 = System.Text.Json.JsonSerializer.Deserialize<ApprovalInfoView>(stringResponse1,
  968. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  969. if (approvalInfoView11.errcode != 0)
  970. {
  971. return approvalInfoView1;
  972. }
  973. new_cursor = approvalInfoView11.new_next_cursor;
  974. if (approvalInfoView11.sp_no_list.Count > 0 || approvalInfoView11.sp_no_list != null)
  975. {
  976. sp_no_lists.AddRange(approvalInfoView1.sp_no_list);
  977. }
  978. }
  979. #endregion
  980. if (approvalInfoView1.errcode != 0)
  981. {
  982. return approvalInfoView1;
  983. }
  984. if (approvalInfoView1.sp_no_list.Count > 0 || approvalInfoView1.sp_no_list != null)
  985. {
  986. sp_no_lists.AddRange(approvalInfoView1.sp_no_list);
  987. }
  988. }
  989. approvalInfoView.errcode = 0;
  990. if (sp_no_lists.Count > 0)
  991. {
  992. sp_no_lists = sp_no_lists.Distinct().ToList();
  993. }
  994. approvalInfoView.sp_no_list = sp_no_lists;
  995. sw.Stop();
  996. approvalInfoView.errmsg = approvalInfoView.errmsg + " 耗时:" + sw.Elapsed.TotalMilliseconds + "ms";
  997. return approvalInfoView;
  998. }
  999. /// <summary>
  1000. /// 获取审批申请详情
  1001. /// </summary>
  1002. /// <param name="sp_no"></param>
  1003. /// <returns></returns>
  1004. public async Task<ApprovalDetailView> GetApprovalDetailAsync(string sp_no)
  1005. {
  1006. ApprovalDetailView ApprovalDetailView = new ApprovalDetailView();
  1007. if (string.IsNullOrEmpty(sp_no))
  1008. {
  1009. return ApprovalDetailView;
  1010. }
  1011. //获取审批数据 token
  1012. Access_TokenView access_Token = await GetTokenAsync(5);
  1013. if (access_Token.errcode != 0)
  1014. {
  1015. ApprovalDetailView.errcode = access_Token.errcode;
  1016. ApprovalDetailView.errmsg = string.Format("【企业微信】【审批申请详情】【Token】【Msg】{0}", access_Token.errmsg);
  1017. return ApprovalDetailView;
  1018. }
  1019. string url = string.Format("/cgi-bin/oa/getapprovaldetail?access_token={0}", access_Token.access_token);
  1020. ApprovalDetail_Request approvalDetail_Req = new ApprovalDetail_Request()
  1021. {
  1022. access_token = access_Token.access_token,
  1023. sp_no = sp_no,
  1024. };
  1025. var json = System.Text.Json.JsonSerializer.Serialize(approvalDetail_Req);
  1026. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1027. var create_Req = await _httpClient.PostAsync(url, content);
  1028. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1029. ApprovalDetailView = System.Text.Json.JsonSerializer.Deserialize<ApprovalDetailView>(stringResponse,
  1030. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1031. return ApprovalDetailView;
  1032. }
  1033. /// <summary>
  1034. /// 批量获取审批详情
  1035. /// </summary>
  1036. /// <param name="startDt"></param>
  1037. /// <param name="endDt"></param>
  1038. /// <param name="creator">
  1039. /// 申请人
  1040. /// </param>
  1041. /// <param name="sp_status">
  1042. /// sp_status-审批单状态(1-审批中;2-已通过;3-已驳回;4-已撤销;6-通过后撤销;7-已删除;10-已支付)
  1043. /// </param>
  1044. /// <param name="record_type">
  1045. /// record_type - 审批单类型属性,1-请假;2-打卡补卡;3-出差;4-外出;5-加班; 6- 调班;7-会议室预定;8-退款审批;9-红包报销审批
  1046. /// </param>
  1047. /// <returns></returns>
  1048. public async Task<List<Sp_Detail>> GetApprovalDetailsAsync(DateTime startDt, DateTime endDt, string creator, int sp_status, int record_type)
  1049. {
  1050. Stopwatch sw = new Stopwatch();
  1051. sw.Start();
  1052. List<Sp_Detail> details = new List<Sp_Detail>();
  1053. if (string.IsNullOrEmpty(creator))
  1054. {
  1055. return details;
  1056. }
  1057. ApprovalInfoView approvalInfoView = new ApprovalInfoView();
  1058. approvalInfoView = await GetApprovalInfoAsync(startDt, endDt, creator, sp_status, record_type);
  1059. if (approvalInfoView.errcode != 0)
  1060. {
  1061. return details;
  1062. }
  1063. //循环查询详情
  1064. foreach (var item in approvalInfoView.sp_no_list)
  1065. {
  1066. ApprovalDetailView approvalDetailView = new ApprovalDetailView();
  1067. approvalDetailView = await GetApprovalDetailAsync(item);
  1068. if (approvalDetailView.errcode != 0)
  1069. {
  1070. Log.Error("【GetApprovalDetailsAsync】 record_type:" + record_type + " ErrorMsg:" + approvalDetailView.errmsg);
  1071. break;
  1072. }
  1073. details.Add(approvalDetailView.info);
  1074. }
  1075. sw.Stop();
  1076. double hs = sw.Elapsed.TotalMilliseconds;
  1077. return details;
  1078. }
  1079. /// <summary>
  1080. /// 获取审批模板详情
  1081. /// </summary>
  1082. /// <param name="template_id">模板Id</param>
  1083. /// <returns></returns>
  1084. public async Task<TemplateDetailView> GetTemplateDetailAsync(string template_id)
  1085. {
  1086. TemplateDetailView templateDetailView = new TemplateDetailView();
  1087. if (string.IsNullOrEmpty(template_id))
  1088. {
  1089. templateDetailView.errmsg = "template_id为空!";
  1090. return templateDetailView;
  1091. }
  1092. //获取审批数据 token
  1093. Access_TokenView access_Token = await GetTokenAsync(5);
  1094. if (access_Token.errcode != 0)
  1095. {
  1096. templateDetailView.errcode = access_Token.errcode;
  1097. templateDetailView.errmsg = string.Format("【企业微信】【审批申请详情】【Token】【Msg】{0}", access_Token.errmsg);
  1098. return templateDetailView;
  1099. }
  1100. string url = string.Format("/cgi-bin/oa/gettemplatedetail?access_token={0}", access_Token.access_token);
  1101. var approvalDetail_Req = new
  1102. {
  1103. access_token = access_Token.access_token,
  1104. template_id = template_id,
  1105. };
  1106. var json = System.Text.Json.JsonSerializer.Serialize(approvalDetail_Req);
  1107. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1108. var create_Req = await _httpClient.PostAsync(url, content);
  1109. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1110. templateDetailView = System.Text.Json.JsonSerializer.Deserialize<TemplateDetailView>(stringResponse,
  1111. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1112. return templateDetailView;
  1113. }
  1114. #endregion
  1115. #region hook机器人
  1116. /// <summary>
  1117. /// 发送团组信息给财务群
  1118. /// </summary>
  1119. /// <param name="msg"></param>
  1120. /// <returns></returns>
  1121. /// <exception cref="NotImplementedException"></exception>
  1122. public async Task<ResponseBase> RobotSendMsg_GroupInfo(string msg)
  1123. {
  1124. string key = "b4fe152f-a97a-48b1-830f-ab447f6d2f5f";
  1125. string url = string.Format("/cgi-bin/webhook/send?key={0}", key);
  1126. RobotSendMsg_GroupInfo_Content contentStr = new RobotSendMsg_GroupInfo_Content() { content = msg };
  1127. RobotSendMsg_GroupInfo_Param reqJson = new RobotSendMsg_GroupInfo_Param()
  1128. {
  1129. msgtype = "markdown",
  1130. markdown = contentStr
  1131. };
  1132. var json = System.Text.Json.JsonSerializer.Serialize(reqJson);
  1133. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1134. var create_Req = await _httpClient.PostAsync(url, content);
  1135. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1136. ResponseBase sendMsgView = System.Text.Json.JsonSerializer.Deserialize<ResponseBase>(stringResponse,
  1137. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1138. return sendMsgView;
  1139. }
  1140. #endregion
  1141. #region 团组状态通知
  1142. /// <summary>
  1143. /// 获取部门成员
  1144. /// </summary>
  1145. /// <param name="departmentId"></param>
  1146. /// <returns></returns>
  1147. public async Task<GroupStatus_UserSimplelistView> GroupStatus_GetUserList(int departmentId = 8402038)
  1148. {
  1149. GroupStatus_UserSimplelistView result = new GroupStatus_UserSimplelistView();
  1150. Access_TokenView access_Token = await GetTokenAsync(6);
  1151. if (access_Token.errcode != 0)
  1152. {
  1153. result.errcode = access_Token.errcode;
  1154. result.errmsg = string.Format("【企业微信】【团组状态通知】【Token】【Msg】{0}", access_Token.errmsg);
  1155. return result;
  1156. }
  1157. string url = string.Format("/cgi-bin/user/simplelist?access_token={0}&department_id={1}", access_Token.access_token, departmentId);
  1158. var simplelist_Req = await _httpClient.GetAsync(url);
  1159. if (simplelist_Req.IsSuccessStatusCode)
  1160. {
  1161. var stringResponse = await simplelist_Req.Content.ReadAsStringAsync();
  1162. result = System.Text.Json.JsonSerializer.Deserialize<GroupStatus_UserSimplelistView>(stringResponse,
  1163. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1164. }
  1165. else
  1166. {
  1167. result.errcode = -1;
  1168. result.errmsg = "企业微信部门列表未获取到!";
  1169. }
  1170. return result;
  1171. }
  1172. /// <summary>
  1173. /// 创建群聊
  1174. /// </summary>
  1175. /// <param name="chatName"></param>
  1176. /// <param name="owner"></param>
  1177. /// <param name="userList"></param>
  1178. /// <param name="chatId"></param>
  1179. /// <returns></returns>
  1180. public async Task<GroupStatus_CreateChatView> GroupStatus_CreateChat(string chatName, string owner, List<string> userList, string chatId)
  1181. {
  1182. GroupStatus_CreateChatView result = new GroupStatus_CreateChatView();
  1183. Access_TokenView access_Token = await GetTokenAsync(6);
  1184. if (access_Token.errcode != 0)
  1185. {
  1186. result.errcode = access_Token.errcode;
  1187. result.errmsg = string.Format("【企业微信】【团组状态通知】【Token】【Msg】{0}", access_Token.errmsg);
  1188. return result;
  1189. }
  1190. GroupStatus_CreateChat reqJson = new GroupStatus_CreateChat()
  1191. {
  1192. chatid = chatId,
  1193. name = chatName,
  1194. owner = owner,
  1195. userlist = userList
  1196. };
  1197. string url = string.Format("/cgi-bin/appchat/create?access_token={0}&debug=1", access_Token.access_token);
  1198. var json = System.Text.Json.JsonSerializer.Serialize(reqJson);
  1199. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1200. var create_Req = await _httpClient.PostAsync(url, content);
  1201. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1202. result = System.Text.Json.JsonSerializer.Deserialize<GroupStatus_CreateChatView>(stringResponse,
  1203. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1204. return result;
  1205. }
  1206. /// <summary>
  1207. /// 向群聊发送信息
  1208. /// </summary>
  1209. /// <param name="chatId"></param>
  1210. /// <param name="msgContent"></param>
  1211. /// <param name="msgType"></param>
  1212. /// <returns></returns>
  1213. /// <exception cref="NotImplementedException"></exception>
  1214. public async Task<ResponseBase> GroupStatus_SendChatMsg_Markdown(string chatId, string msgContent, string msgType = "markdown")
  1215. {
  1216. ResponseBase result = new ResponseBase();
  1217. Access_TokenView access_Token = await GetTokenAsync(6);
  1218. if (access_Token.errcode != 0)
  1219. {
  1220. result.errcode = access_Token.errcode;
  1221. result.errmsg = string.Format("【企业微信】【团组状态通知】【Token】【Msg】{0}", access_Token.errmsg);
  1222. return result;
  1223. }
  1224. RobotSendMsg_GroupInfo_Content markdown = new RobotSendMsg_GroupInfo_Content() { content = msgContent };
  1225. GroupStatus_AppChatSend_Markdown reqJson = new GroupStatus_AppChatSend_Markdown()
  1226. {
  1227. chatid = chatId,
  1228. msgtype = msgType,
  1229. markdown = markdown
  1230. };
  1231. string url = string.Format("/cgi-bin/appchat/send?access_token={0}", access_Token.access_token);
  1232. var json = System.Text.Json.JsonSerializer.Serialize(reqJson);
  1233. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1234. var create_Req = await _httpClient.PostAsync(url, content);
  1235. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1236. result = System.Text.Json.JsonSerializer.Deserialize<ResponseBase>(stringResponse,
  1237. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1238. return result;
  1239. }
  1240. /// <summary>
  1241. /// 向用户发送Markdown信息
  1242. /// </summary>
  1243. /// <param name="toUser"></param>
  1244. /// <param name="msgContent"></param>
  1245. /// <param name="msgType"></param>
  1246. /// <returns></returns>
  1247. /// <exception cref="NotImplementedException"></exception>
  1248. public async Task<GroupStatus_SendMessageView> GroupStatus_SendMessage_ToUser_Markdown(List<string> toUser, string msgContent, string msgType = "markdown")
  1249. {
  1250. GroupStatus_SendMessageView result = new GroupStatus_SendMessageView();
  1251. Access_TokenView access_Token = await GetTokenAsync(6);
  1252. if (access_Token.errcode != 0)
  1253. {
  1254. result.errcode = access_Token.errcode;
  1255. result.errmsg = string.Format("【企业微信】【团组状态通知】【Token】【Msg】{0}", access_Token.errmsg);
  1256. return result;
  1257. }
  1258. RobotSendMsg_GroupInfo_Content markdown = new RobotSendMsg_GroupInfo_Content() { content = msgContent };
  1259. string userIds = "";
  1260. toUser.ForEach(s => userIds += s + "|");
  1261. userIds = userIds.TrimEnd('|');
  1262. GroupStatus_AppSend_Markdown reqJson = new GroupStatus_AppSend_Markdown()
  1263. {
  1264. markdown = markdown,
  1265. touser = userIds
  1266. };
  1267. string url = string.Format("/cgi-bin/message/send?access_token={0}&debug=1", access_Token.access_token);
  1268. var json = System.Text.Json.JsonSerializer.Serialize(reqJson);
  1269. var content = new StringContent(json, Encoding.UTF8, "application/json");
  1270. var create_Req = await _httpClient.PostAsync(url, content);
  1271. var stringResponse = await create_Req.Content.ReadAsStringAsync();
  1272. result = System.Text.Json.JsonSerializer.Deserialize<GroupStatus_SendMessageView>(stringResponse,
  1273. new JsonSerializerOptions() { PropertyNamingPolicy = JsonNamingPolicy.CamelCase });
  1274. return result;
  1275. }
  1276. #endregion
  1277. public long ConvertToTimeSpan(DateTime dt)
  1278. {
  1279. return (dt.ToUniversalTime().Ticks - 621355968000000000) / 10000000;
  1280. }
  1281. }
  1282. }