CoreAPIHttpService.cs 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903
  1. using DocumentFormat.OpenXml.Wordprocessing;
  2. using HTEXLib.COMM.Helpers;
  3. using HTEXLib.Helpers.ShapeHelpers;
  4. using Microsoft.AspNetCore.Hosting;
  5. using Microsoft.Extensions.Configuration;
  6. using Microsoft.Extensions.DependencyInjection;
  7. using Microsoft.Extensions.Hosting;
  8. using Microsoft.Extensions.Logging;
  9. using Microsoft.Extensions.Options;
  10. using Newtonsoft.Json;
  11. using OpenXmlPowerTools;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Configuration;
  15. using System.Dynamic;
  16. using System.IO;
  17. using System.Linq;
  18. using System.Net;
  19. using System.Net.Http;
  20. using System.Net.Http.Headers;
  21. using System.Net.Http.Json;
  22. using System.Security.Policy;
  23. using System.Text;
  24. using System.Text.Json;
  25. using System.Threading.Tasks;
  26. using System.Web;
  27. using TEAMModelOS.Models;
  28. using TEAMModelOS.SDK.DI;
  29. using TEAMModelOS.SDK.Extension;
  30. using TEAMModelOS.SDK.Models;
  31. using TEAMModelOS.SDK.Models.Cosmos.BI;
  32. using TEAMModelOS.SDK.Models.Service;
  33. using static TEAMModelOS.SDK.CoreAPIHttpService;
  34. namespace TEAMModelOS.SDK
  35. {
  36. public class CoreAPIHttpServiceOptions
  37. {
  38. public string location { get; set; }
  39. public string coreUrl { get; set; }
  40. public string bbUrl { get; set; }
  41. public string clientID { get; set; }
  42. public string clientSecret { get; set; }
  43. }
  44. public class CoreAPIHttpService
  45. {
  46. //private readonly IWebHostEnvironment _environment;
  47. public bool check=true;
  48. private SnowflakeId _snowflakeId;
  49. private readonly HttpClient _httpClient;
  50. public readonly CoreAPIHttpServiceOptions options;
  51. //private readonly IConfiguration _configuration;
  52. // private readonly DI.DingDing _dingDing;
  53. //public readonly Option _option;
  54. public CoreAPIHttpService(HttpClient httpClient, IConfiguration _configuration , SnowflakeId snowflakeId)
  55. {
  56. _httpClient = httpClient;
  57. string location = _configuration.GetValue<string>("Option:Location");
  58. string url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  59. string bburl = _configuration.GetValue<string>("HaBookAuth:CoreBBAPI");
  60. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  61. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  62. options = new CoreAPIHttpServiceOptions
  63. {
  64. location=location,
  65. coreUrl=url,
  66. bbUrl=bburl,
  67. clientID=clientID,
  68. clientSecret=clientSecret
  69. };
  70. //_environment = environment;
  71. _snowflakeId = snowflakeId;
  72. //_option = option?.Value;
  73. //_configuration = configuration;
  74. //_dingDing = dingDing;
  75. }
  76. public class CoreAPIToken {
  77. public string id_token { get; set; }
  78. public string access_token { get; set; }
  79. public string expires_in { get; set; }
  80. public string token_type { get; set; }
  81. }
  82. public async Task<(HttpStatusCode code,string msg )> AdminChange(AdminChange data, string location, IConfiguration _configuration, DI.DingDing _dingDing)
  83. {
  84. try
  85. {
  86. string url = _configuration.GetValue<string>("HaBookAuth:CoreBBAPI") + "/ies5/admin-change";
  87. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  88. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  89. if (location.Contains("China"))
  90. {
  91. location = "China";
  92. }
  93. else if (location.Contains("Global"))
  94. {
  95. location = "Global";
  96. }
  97. var client = _httpClient;
  98. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  99. if (client.DefaultRequestHeaders.Contains("Authorization"))
  100. {
  101. client.DefaultRequestHeaders.Remove("Authorization");
  102. }
  103. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  104. HttpResponseMessage responseMessage = await client.PostAsJsonAsync(url, data);
  105. if (responseMessage.Content != null)
  106. {
  107. string content = await responseMessage.Content.ReadAsStringAsync();
  108. if (!string.IsNullOrWhiteSpace(content))
  109. {
  110. return (responseMessage.StatusCode, content);
  111. }
  112. else
  113. {
  114. return (responseMessage.StatusCode, null);
  115. }
  116. }
  117. else
  118. {
  119. return (responseMessage.StatusCode, null);
  120. }
  121. }
  122. catch (Exception ex)
  123. {
  124. await _dingDing.SendBotMsg($"{location},管理员或管理权限变更通知BB处理相关通知异常\n{data.ToJsonString()}{ex.Message},\n{ex.StackTrace}", GroupNames.成都开发測試群組);
  125. return (HttpStatusCode.InternalServerError, null);
  126. }
  127. }
  128. public async Task<(HttpStatusCode code , CoreAPIToken token )> GetCoreAPIoAuth2Token(Dictionary<string,object> data, string location, IConfiguration _configuration, DI.DingDing _dingDing) {
  129. try
  130. {
  131. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  132. //coreUrl = "https://api2-rc.teammodel.cn";
  133. url = $"{url}/oauth2/token";
  134. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  135. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  136. if (location.Contains("China"))
  137. {
  138. location = "China";
  139. }
  140. else if (location.Contains("Global"))
  141. {
  142. location = "Global";
  143. }
  144. var client = _httpClient ;
  145. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  146. if (client.DefaultRequestHeaders.Contains("Authorization"))
  147. {
  148. client.DefaultRequestHeaders.Remove("Authorization");
  149. }
  150. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  151. HttpResponseMessage responseMessage = await client.PostAsJsonAsync(url, data);
  152. if (responseMessage.Content != null)
  153. {
  154. string content = await responseMessage.Content.ReadAsStringAsync();
  155. if (!string.IsNullOrWhiteSpace(content))
  156. {
  157. CoreAPIToken coreAPI = content.ToObject<CoreAPIToken>();
  158. return (responseMessage.StatusCode, coreAPI);
  159. }
  160. else
  161. {
  162. return (responseMessage.StatusCode, null);
  163. }
  164. }
  165. else
  166. {
  167. return (responseMessage.StatusCode, null);
  168. }
  169. }
  170. catch (Exception ex)
  171. {
  172. await _dingDing.SendBotMsg($"{location}验证码发送异常:\n{ex.Message}\n{ex.StackTrace}", DI.GroupNames.醍摩豆服務運維群組);
  173. return (HttpStatusCode.InternalServerError, null);
  174. }
  175. }
  176. /*
  177. hubName string Optional 指定要傳送到哪個訊息中樞,若沒給則不傳送端外通知,只會發送端內通知。(目前只有"hita"及"hita5"能使用)
  178. sender string Optional 發送訊息的來源端
  179. tags string array Required 接收對象或手機註冊的tag,ID或服務等...
  180. title string Optional 標題,會"直接"顯示於端外通知的title位置
  181. body string Optional 正文,會"直接"顯示於端外通知的body位置
  182. eventId string Optional 事件ID
  183. eventName string Optional 事件名稱
  184. data string Optional 額外資料
  185. 'request_school': vm.$t('notice.type1'), // 管理员收到他人申请加入的通知
  186. 'invite_school': vm.$t('notice.type2'), // 你收到学校邀请你的通知
  187. 'remove_school': vm.$t('notice.type3'), // 学校将你移除的通知
  188. 'request-join_school': vm.$t('notice.type1'), // 学校同意你的加入请求
  189. 'invite-join_school': vm.$t('notice.type2'), // 某人已同意你对他的邀请
  190. 'coedit_syllabus': vm.$t('notice.type4'), // 邀请课纲共编的通知
  191. 'share_syllabus': vm.$t('notice.type4'), // 个人课纲分享的通知
  192. 'transfer-admin_school': vm.$t('notice.type5'), // 管理员转移的通知
  193. 'scoring-arb_school': vm.$t('notice.type6'), // 仲裁卷阅卷任务分配通知
  194. 'scoring-err_school': vm.$t('notice.type6'), // 异常卷阅卷任务分配通知
  195. 'scoring-mark_school': vm.$t('notice.type6'), // 普通阅卷任务分配通知
  196. 'scan-join_groupList': vm.$t('notice.type7'), // 扫码加入名单通知
  197. 'scan-join_school': vm.$t('notice.type7'), // 扫码加入学校通知
  198. 'submitanswer_homework': vm.$t('notice.type8'), // 作业提交通知
  199. 'expire_lessonRecord': vm.$t('notice.type9'), // 课例过期通知
  200. */
  201. /// <summary>
  202. ///
  203. /// 發送端內外通知
  204. ///
  205. /// id: 教师id,name 教师名称,code 语系
  206. /// </summary>
  207. /// <param name="toTeachers"></param>
  208. /// <param name="notifyCode"></param>
  209. /// <param name="replaceData"></param>
  210. /// <param name="location"></param>
  211. /// <param name="_configuration"></param>
  212. /// <param name="_dingDing"></param>
  213. /// <returns></returns>
  214. public void PushNotify(List<IdNameCode> toTeachers , string notifyCode,string notifyType,Dictionary<string, object> replaceData,
  215. string location, IConfiguration _configuration, DI.DingDing _dingDing,string rootPath) {
  216. /*
  217. * IES5_Management shift-assist_school DelBeforeCopyAbility-mark_start copyAbility-mark_finish copyAbility-mark_start
  218. * transfer-admin_school invite-join_school invite_school request_school request-join_school remove_school scan-join_school
  219. * IES5_Course submitanswer_homework scan-join_groupList expire-school_lessonRecord
  220. * IES5_Task scoring-mark_school scoring-arb_school scoring-err_school
  221. * IES5_Contect coedit_syllabus share_syllabus
  222. {
  223. "hubName":"hita5",
  224. "sender":"ies5",
  225. "tags":["1595321354_IES5_Management"]
  226. "title":"",
  227. "body":"",
  228. "eventId":"",
  229. "eventName":"",
  230. "data":""action":{"type":"click\link","tokenbindtype":1,"url":"http://xxxx"}"
  231. }
  232. {
  233. "action":[ //陣列, 可以包含多個按鈕
  234. {
  235. "type":"click", //按鈕
  236. "label":"Ok", //按鈕的字
  237. "url":"https://www.teammodel.net?code=", // POST API
  238. "tokenbindtype":1 //附掛 code token到 utl 後面
  239. }
  240. ]
  241. }
  242. {
  243. "action": [
  244. {
  245. "type": "click",
  246. "label": "\\u67e5\\u770b",
  247. "url": "https:\\/\\/sokrates.teammodel.org\\/exhibition\\/tbavideo\\/check-with-habook\\/?to=aHR0cHM6Ly9zb2tyYXRlcy50ZWFtbW9kZWwub3JnL2V4aGliaXRpb24vdGJhdmlkZW8jL2NvbnRlbnQvMTM5MzU\\/Z3JvdXBJZHM9MTQxJmNoYW5uZWxJZD0xMzA=&ticket=",
  248. "tokenbindtype": 1
  249. }
  250. ]
  251. }
  252. */
  253. try
  254. {
  255. string lang = location.Contains("China") ? "zh-cn" : "en-us";
  256. toTeachers.FindAll(x => string.IsNullOrWhiteSpace(x.code) || (!x.code.Equals("zh-cn") && !x.code.Equals("zh-tw") && !x.code.Equals("en-us"))).ForEach(x => { x.code = lang; });
  257. var groups = toTeachers.GroupBy(x => x.code).Select(x => new { x.Key, list = x.ToList() });
  258. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  259. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  260. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  261. string site = location;
  262. if (location.Contains("China"))
  263. {
  264. location = "China";
  265. }
  266. else if (location.Contains("Global"))
  267. {
  268. location = "Global";
  269. }
  270. replaceData.Add("notifyCode", notifyCode);
  271. replaceData.Add("location", site);
  272. if (replaceData.ContainsKey("schoolId"))
  273. {
  274. replaceData.Add("scope", "school");
  275. }
  276. else { replaceData.Add("scope", "private"); }
  277. var client = _httpClient;
  278. var token = CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location).Result;
  279. if (client.DefaultRequestHeaders.Contains("Authorization"))
  280. {
  281. client.DefaultRequestHeaders.Remove("Authorization");
  282. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  283. }
  284. else
  285. {
  286. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  287. }
  288. foreach (var group in groups)
  289. {
  290. string path =Path.Combine(rootPath, $"Lang/{group.Key}.json");
  291. var sampleJson = File.ReadAllBytes(path).AsSpan();
  292. Utf8JsonReader reader = new Utf8JsonReader(sampleJson);
  293. if (JsonDocument.TryParseValue(ref reader, out JsonDocument jsonDoc) && jsonDoc.RootElement.TryGetProperty(notifyCode, out JsonElement json))
  294. {
  295. List<string> msgs = json.ToObject<List<string>>();
  296. List<dynamic> actions = new List<dynamic>();
  297. if (msgs.IsNotEmpty())
  298. {
  299. dynamic action = new ExpandoObject();
  300. //处理action
  301. string urlAction = "";
  302. if (notifyCode.Equals("request_school") || notifyCode.Equals("invite_school")/*|| notifyCode.Equals("quit_school")*/) {
  303. if (site.Equals("Global")) {
  304. urlAction = "https://www.teammodel.net/core/process-notify";
  305. }
  306. if (site.Equals("Global-Test")|| site.Equals("Global-Dep"))
  307. {
  308. urlAction = "https://test.teammodel.net/core/process-notify";
  309. }
  310. if (site.Equals("China"))
  311. {
  312. urlAction = "https://www.teammodel.cn/core/process-notify";
  313. }
  314. if (site.Equals("China-Test")|| site.Equals("China-Dep"))
  315. {
  316. urlAction = "https://test.teammodel.cn/core/process-notify";
  317. }
  318. byte[] byts = Encoding.Unicode.GetBytes(replaceData.ToJsonString());
  319. var rdata=Convert.ToBase64String(byts);
  320. byte[] bs= Convert.FromBase64String(rdata);
  321. string data1 = Encoding.Unicode.GetString(bs);
  322. rdata = HttpUtility.UrlEncode(rdata, Encoding.UTF8);
  323. urlAction = $"{urlAction}?notifyCode={notifyCode}&data={rdata}";
  324. if (msgs.Count == 3)
  325. {
  326. string urlA = $"{urlAction}&notifyEvent=1&ticket=";
  327. actions.Add(new { type = "click", label = msgs[2], url = urlA, tokenbindtype = 1 });
  328. }
  329. if (msgs.Count == 4) {
  330. string urlA = $"{urlAction}&notifyEvent=1&ticket=";
  331. string urlB = $"{urlAction}&notifyEvent=2&ticket=";
  332. actions.Add(new { type = "click", label = msgs[2], url = urlA, tokenbindtype = 1 });
  333. actions.Add(new { type = "click", label = msgs[3], url = urlB, tokenbindtype = 1 });
  334. }
  335. }
  336. var tags = group.list.Select(x => $"{x.id}_{notifyType}");
  337. string data = new { value = replaceData }.ToJsonString();
  338. if (actions.Any()) {
  339. data=new { value = replaceData ,action=actions}.ToJsonString();
  340. }
  341. NotifyData notifyData = new NotifyData
  342. {
  343. hubName = "hita5",
  344. sender = "IES",
  345. tags = tags.ToList(),
  346. title = msgs[0],
  347. eventId = $"{notifyCode}-{_snowflakeId.NextId()}",
  348. eventName = $"{msgs[0]}",
  349. data = data
  350. };
  351. if (msgs.Count == 1)
  352. {
  353. notifyData.body = msgs[0];
  354. }
  355. else
  356. {
  357. replaceData.Keys.ToList().ForEach(x => {
  358. msgs[1] = msgs[1].Replace("{" + x + "}", $"{replaceData[x]}");
  359. });
  360. notifyData.body = msgs[1];
  361. }
  362. string result = "";
  363. HttpResponseMessage responseMessage = _httpClient.PostAsJsonAsync($"{url}/service/PushNotify", notifyData).Result;
  364. if (responseMessage.StatusCode == HttpStatusCode.OK)
  365. {
  366. string content = responseMessage.Content.ReadAsStringAsync().Result;
  367. result = content;
  368. }
  369. else
  370. {
  371. result = $"{responseMessage.StatusCode},推送返回的状态码。";
  372. }
  373. // _dingDing.SendBotMsg($"{location}站点发送消息:\n{url}/service/PushNotify \nheader: {token.AccessToken} \nresult:{result}\n params:{notifyData.ToJsonString()}", GroupNames.成都开发測試群組).GetAwaiter().GetResult();
  374. }
  375. }
  376. }
  377. } catch (Exception ex) {
  378. _ = _dingDing.SendBotMsg($"{location},通知V2发送异常\n{ex.Message}\n{ex.StackTrace}\n{replaceData.ToJsonString()}\n{toTeachers.ToJsonString()}\n{notifyCode}", GroupNames.醍摩豆服務運維群組);
  379. }
  380. }
  381. /// <summary>
  382. /// BI发送端外通知消息
  383. /// </summary>
  384. /// <param name="bINotice"></param>
  385. /// <param name="notifyCode"></param>
  386. /// <param name="notifyType"></param>
  387. /// <param name="replaceData"></param>
  388. /// <param name="location"></param>
  389. /// <param name="_configuration"></param>
  390. /// <param name="_dingDing"></param>
  391. public void BIPushNotify(BINotice bINotice, Dictionary<string, object> replaceData, string location, IConfiguration _configuration, DingDing _dingDing)
  392. {
  393. try
  394. {
  395. string notifyCode = "";
  396. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  397. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  398. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  399. string site = location;
  400. string sendSite = "";
  401. //处理action
  402. string urlAction = "";
  403. List<dynamic> actions = new List<dynamic>();
  404. if (site.Equals("Global"))
  405. {
  406. sendSite = "Global";
  407. urlAction = "https://bi-rc.teammodel.net/core/process-notify";
  408. }
  409. if (site.Equals("Global-Test") || site.Equals("Global-Dep"))
  410. {
  411. sendSite = "Global";
  412. urlAction = "https://bi-rc.teammodel.net/core/process-notify";
  413. }
  414. if (site.Equals("China"))
  415. {
  416. sendSite = "China";
  417. urlAction = "https://bi.teammodel.cn/core/process-notify";
  418. }
  419. if (site.Equals("China-Test") || site.Equals("China-Dep"))
  420. {
  421. sendSite = "China";
  422. urlAction = "https://bitest.teammodel.cn/core/process-notify";
  423. //urlAction = "https://localhost:5001/core/process-notify";
  424. }
  425. if (!string.IsNullOrEmpty($"{bINotice.jumpUrl}"))
  426. {
  427. urlAction = bINotice.jumpUrl;
  428. }
  429. if (!string.IsNullOrEmpty($"{bINotice.callbackName}") && !string.IsNullOrEmpty($"{bINotice.refuseName}"))
  430. {
  431. notifyCode = "notify_agree";
  432. }
  433. else if (!string.IsNullOrEmpty($"{bINotice.callbackName}") && string.IsNullOrEmpty($"{bINotice.refuseName}"))
  434. {
  435. notifyCode = "notify_see";
  436. }
  437. else
  438. {
  439. notifyCode = "notify_default";
  440. }
  441. replaceData.Add("noticeId", bINotice.id);
  442. replaceData.Add("notifyCode", notifyCode);
  443. replaceData.Add("location", site);
  444. byte[] byts = Encoding.Unicode.GetBytes(replaceData.ToJsonString());
  445. var rdata = Convert.ToBase64String(byts);
  446. byte[] bs = Convert.FromBase64String(rdata);
  447. string data1 = Encoding.Unicode.GetString(bs);
  448. rdata = HttpUtility.UrlEncode(rdata, Encoding.UTF8);
  449. urlAction = $"{urlAction}?notifyCode={bINotice.createId}&data={rdata}";
  450. if (!string.IsNullOrEmpty($"{bINotice.callbackName}")&& !string.IsNullOrEmpty($"{bINotice.refuseName}"))
  451. {
  452. //string id_token = "eyJhbGciOiJodHRwOi8vd3d3LnczLm9yZy8yMDAxLzA0L3htbGRzaWctbW9yZSNobWFjLXNoYTI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJsb2NhbGhvc3Q6NTAwMSIsInN1YiI6IjE2MzYwMTY0OTkiLCJhenAiOnRydWUsImV4cCI6IjE2NzU3NTYyOTciLCJuYW1lIjoi5b2t56S8IiwicGljdHVyZSI6bnVsbCwicm9sZXMiOlsiYWRtaW4iXSwicGVybWlzc2lvbnMiOlsiYXJlYWRhdGEtcmVhZCIsImFyZWFkYXRhLXVwZCIsInNjaG9vbGRhdGEtcmVhZCIsInNjaG9vbGRhdGEtdXBkIiwiY2hpbmEtcmVhZCIsImdsb2JhbC1yZWFkIl0sImRkc3ViIjoiMTYzMjk5MzA5MjMxNjc0NDEiLCJkZG5hbWUiOiLlva3npLwiLCJkZHBpY3R1cmUiOiIiLCJzY29wZSI6ImFzc2lzdCIsIndlYnNpdGUiOiJDaGluYSJ9.J_MCukBvxB9N0voo7uh9o2wK_3g8w-Dw7zbK95GanT4";
  453. //string urlA = $"{urlAction}&notifyEvent=1&token={id_token}&ticket=";
  454. string urlA = $"{urlAction}&notifyEvent=1&ticket=";
  455. string urlB = $"{urlAction}&notifyEvent=2&ticket=";
  456. actions.Add(new { type = "click", label = bINotice.callbackName, url = urlA, tokenbindtype = 1 });
  457. actions.Add(new { type = "click", label = bINotice.refuseName, url = urlB, tokenbindtype = 1 });
  458. }
  459. else if (!string.IsNullOrEmpty($"{bINotice.callbackName}") && string.IsNullOrEmpty($"{ bINotice.refuseName}"))
  460. {
  461. string urlA = $"{urlAction}&notifyEvent=1&ticket=";
  462. actions.Add(new { type = "click", label = bINotice.callbackName, url = urlA, tokenbindtype = 1 });
  463. }
  464. var client = _httpClient;
  465. var token = CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, sendSite).Result;
  466. if (client.DefaultRequestHeaders.Contains("Authorization"))
  467. {
  468. client.DefaultRequestHeaders.Remove("Authorization");
  469. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  470. }
  471. else
  472. {
  473. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  474. }
  475. string data = new { notice = bINotice, value = replaceData }.ToJsonString();
  476. if (actions.Any())
  477. {
  478. data = new { value = replaceData, action = actions }.ToJsonString();
  479. }
  480. NotifyData notifyData = new()
  481. {
  482. hubName = string.Join(",", bINotice.crowd.types),
  483. sender = bINotice.source,
  484. tags = bINotice.crowdIds,
  485. title = bINotice.theme,
  486. body = bINotice.content,
  487. eventId = $"BI-send",
  488. eventName = $"BI_SendNotice",
  489. data = data
  490. };
  491. string result = "";
  492. HttpResponseMessage responseMessage = _httpClient.PostAsJsonAsync($"{url}/service/PushNotify", notifyData).Result;
  493. if (responseMessage.StatusCode == HttpStatusCode.OK)
  494. {
  495. string content = responseMessage.Content.ReadAsStringAsync().Result;
  496. result = content;
  497. }
  498. else
  499. result = $"{responseMessage.StatusCode},推送返回的状态码。";
  500. }
  501. catch (Exception ex)
  502. {
  503. _ = _dingDing.SendBotMsg($"{location},BI通知V2发送异常\n{ex.Message}\n{ex.StackTrace}\n{replaceData.ToJsonString()}\n{bINotice.ToJsonString()}", GroupNames.成都开发測試群組);
  504. }
  505. }
  506. /// <summary>
  507. /// 每日课程消息发送
  508. /// </summary>
  509. /// <param name="notifyData"></param>
  510. /// <param name="replaceData"></param>
  511. /// <param name="location"></param>
  512. /// <param name="_configuration"></param>
  513. /// <param name="_dingDing"></param>
  514. public void CoursePushNotify(NotifyData notifyData, Dictionary<string, object> replaceData, string location, IConfiguration _configuration, DingDing _dingDing)
  515. {
  516. try
  517. {
  518. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  519. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  520. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  521. string site = location;
  522. string sendSite = "";
  523. //处理action
  524. string urlAction = "";
  525. List<dynamic> actions = new List<dynamic>();
  526. if (site.Equals("Global"))
  527. {
  528. sendSite = "Global";
  529. urlAction = "https://www.teammodel.net/core/process-notify";
  530. }
  531. if (site.Equals("Global-Test") || site.Equals("Global-Dep"))
  532. {
  533. sendSite = "Global";
  534. urlAction = "https://test.teammodel.net/core/process-notify";
  535. }
  536. if (site.Equals("China"))
  537. {
  538. sendSite = "China";
  539. urlAction = "https://www.teammodel.cn/core/process-notify";
  540. }
  541. if (site.Equals("China-Test") || site.Equals("China-Dep"))
  542. {
  543. sendSite = "China";
  544. urlAction = "https://test.teammodel.cn/core/process-notify";
  545. }
  546. var client = _httpClient;
  547. var token = CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, sendSite).Result;
  548. if (client.DefaultRequestHeaders.Contains("Authorization"))
  549. {
  550. client.DefaultRequestHeaders.Remove("Authorization");
  551. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  552. }
  553. else
  554. {
  555. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  556. }
  557. string data = new { value = replaceData }.ToJsonString();
  558. if (actions.Any())
  559. {
  560. data = new { value = replaceData, action = actions }.ToJsonString();
  561. }
  562. string result = "";
  563. HttpResponseMessage responseMessage = _httpClient.PostAsJsonAsync($"{url}/service/PushNotify", notifyData).Result;
  564. if (responseMessage.StatusCode == HttpStatusCode.OK)
  565. {
  566. string content = responseMessage.Content.ReadAsStringAsync().Result;
  567. result = content;
  568. }
  569. else
  570. result = $"{responseMessage.StatusCode},推送返回的状态码。";
  571. }
  572. catch (Exception ex)
  573. {
  574. _ = _dingDing.SendBotMsg($"{location},BI通知V2发送异常\n{ex.Message}\n{ex.StackTrace}\n{replaceData.ToJsonString()}\n{notifyData.ToJsonString()}", GroupNames.成都开发測試群組);
  575. }
  576. }
  577. public class NotifyData {
  578. public string hubName { get; set; }
  579. public string sender { get; set; }
  580. public List<string> tags { get; set; } = new List<string>();
  581. public string title { get; set; }
  582. public string body { get; set; }
  583. public string eventId { get; set; }
  584. public string eventName { get; set; }
  585. public string data { get; set; }
  586. }
  587. /// <summary>
  588. /// 发送短信验证码
  589. /// </summary>
  590. /// <param name="clientID"></param>
  591. /// <param name="clientSecret"></param>
  592. /// <param name="location"></param>
  593. /// <param name="url"></param>
  594. /// <param name="data"></param>
  595. /// <returns></returns>
  596. public async Task<(HttpStatusCode code ,string content)> SendSmsPin(Dictionary<string, object> data, string location, IConfiguration _configuration, DI.DingDing _dingDing)
  597. {
  598. try {
  599. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  600. //url = "https://api2-rc.teammodel.cn";
  601. url = $"{url}/service/sandsms/pin";
  602. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  603. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  604. if (location.Contains("China"))
  605. {
  606. location = "China";
  607. }
  608. else if (location.Contains("Global"))
  609. {
  610. location = "Global";
  611. }
  612. var client = _httpClient;
  613. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  614. if (client.DefaultRequestHeaders.Contains("Authorization"))
  615. {
  616. client.DefaultRequestHeaders.Remove("Authorization");
  617. }
  618. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  619. HttpResponseMessage responseMessage = await client.PostAsJsonAsync(url, data);
  620. if (responseMessage.Content != null)
  621. {
  622. string content = await responseMessage.Content.ReadAsStringAsync();
  623. if (!string.IsNullOrWhiteSpace(content))
  624. {
  625. return (responseMessage.StatusCode, content);
  626. }
  627. else {
  628. return (responseMessage.StatusCode, null);
  629. }
  630. }
  631. else {
  632. return (responseMessage.StatusCode, null);
  633. }
  634. } catch (Exception ex) {
  635. await _dingDing.SendBotMsg($"{location}验证码发送异常:\n{ex.Message}\n{ex.StackTrace}", DI.GroupNames.醍摩豆服務運維群組);
  636. return (HttpStatusCode.InternalServerError,null);
  637. }
  638. }
  639. /// <summary>
  640. /// 隐式登录
  641. /// </summary>
  642. /// <param name="clientID"></param>
  643. /// <param name="clientSecret"></param>
  644. /// <param name="location"></param>
  645. /// <param name="url"></param>
  646. /// <param name="data"></param>
  647. /// <returns></returns>
  648. public async Task<TmdidImplicit> Implicit( Dictionary<string,string> data,string location, IConfiguration _configuration)
  649. {
  650. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  651. //url = "https://api2-rc.teammodel.cn";
  652. url = $"{url}/oauth2/implicit";
  653. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  654. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  655. if (location.Contains("China"))
  656. {
  657. location = "China";
  658. }
  659. else if (location.Contains("Global"))
  660. {
  661. location = "Global";
  662. }
  663. var client = _httpClient;
  664. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  665. if (client.DefaultRequestHeaders.Contains("Authorization"))
  666. {
  667. client.DefaultRequestHeaders.Remove("Authorization");
  668. }
  669. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  670. HttpResponseMessage responseMessage = await client.PostAsJsonAsync(url, data);
  671. if (responseMessage.StatusCode == HttpStatusCode.OK)
  672. {
  673. string content=await responseMessage.Content.ReadAsStringAsync();
  674. if (!string.IsNullOrEmpty(content))
  675. {
  676. TmdidImplicit tmdidImplicit = content.ToObject<TmdidImplicit>();
  677. return tmdidImplicit;
  678. }
  679. else
  680. {
  681. return null;
  682. }
  683. }
  684. else if (responseMessage.StatusCode == HttpStatusCode.Unauthorized)
  685. {
  686. return null;
  687. }
  688. else
  689. {
  690. return null;
  691. }
  692. }
  693. /// <summary>
  694. /// 获取多个用户信息
  695. /// </summary>
  696. /// <param name="clientID"></param>
  697. /// <param name="clientSecret"></param>
  698. /// <param name="location"></param>
  699. /// <param name="url"></param>
  700. /// <param name="data"></param>
  701. /// <returns></returns>
  702. public async Task<string> GetUserInfos(StringContent scontent, string name = "Default") {
  703. var clientID = options.clientID;
  704. var clientSecret = options.clientSecret;
  705. var location = options.location;
  706. var url = options.coreUrl;
  707. if (location.Contains("China"))
  708. { url =string.IsNullOrWhiteSpace(url)? "https://api2.teammodel.cn":url;
  709. location = "China";
  710. }
  711. else if (location.Contains("Global"))
  712. {
  713. url = string.IsNullOrWhiteSpace(url) ? "https://api2.teammodel.net" : url;
  714. location = "Global";
  715. }
  716. url = $"{url}/oauth2/getuserinfos";
  717. var client = _httpClient;
  718. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  719. if (client.DefaultRequestHeaders.Contains("Authorization"))
  720. {
  721. client.DefaultRequestHeaders.Remove("Authorization");
  722. }
  723. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  724. HttpResponseMessage responseMessage = await client.PostAsync(url, scontent);
  725. if (responseMessage.StatusCode == HttpStatusCode.OK)
  726. {
  727. string content = await responseMessage.Content.ReadAsStringAsync();
  728. if (!string.IsNullOrEmpty(content))
  729. {
  730. return content;
  731. }
  732. else
  733. {
  734. return null;
  735. }
  736. }
  737. else if (responseMessage.StatusCode == HttpStatusCode.Unauthorized)
  738. {
  739. return null;
  740. }
  741. else
  742. {
  743. return null;
  744. }
  745. }
  746. /// <summary>
  747. /// 获取单个用户信息
  748. /// </summary>
  749. /// <param name="clientID"></param>
  750. /// <param name="clientSecret"></param>
  751. /// <param name="location"></param>
  752. /// <param name="url"></param>
  753. /// <param name="data"></param>
  754. /// <returns></returns>
  755. public async Task<CoreUser> GetUserInfo(Dictionary<string, string> data, string location, IConfiguration _configuration)
  756. {
  757. var url = _configuration.GetValue<string>("HaBookAuth:CoreAPI");
  758. //url = "https://api2-rc.teammodel.cn";
  759. url = $"{url}/oauth2/getuserinfo";
  760. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  761. var clientSecret = _configuration.GetValue<string>("HaBookAuth:CoreService:clientSecret");
  762. if (location.Contains("China"))
  763. {
  764. location = "China";
  765. }
  766. else if (location.Contains("Global"))
  767. {
  768. location = "Global";
  769. }
  770. var client = _httpClient;
  771. var token = await CoreTokenExtensions.CreateAccessToken(clientID, clientSecret, location);
  772. if (client.DefaultRequestHeaders.Contains("Authorization"))
  773. {
  774. client.DefaultRequestHeaders.Remove("Authorization");
  775. }
  776. client.DefaultRequestHeaders.Add("Authorization", $"Bearer {token.AccessToken}");
  777. HttpResponseMessage responseMessage = await client.PostAsJsonAsync(url, data);
  778. if (responseMessage.StatusCode == HttpStatusCode.OK)
  779. {
  780. string content = await responseMessage.Content.ReadAsStringAsync();
  781. if (!string.IsNullOrEmpty(content))
  782. {
  783. CoreUserInfo coreUserInfo = content.ToObject<CoreUserInfo>();
  784. if (coreUserInfo != null)
  785. {
  786. bool isActivate = !string.IsNullOrWhiteSpace(coreUserInfo.password) || !string.IsNullOrWhiteSpace(coreUserInfo.passwordOld);
  787. CoreUser coreUser = new CoreUser
  788. {
  789. isActivate = isActivate,
  790. id = coreUserInfo.id,
  791. vid = coreUserInfo.vid,
  792. mail = coreUserInfo.mail,
  793. mobile = coreUserInfo.mobile,
  794. name = coreUserInfo.name,
  795. picture = coreUserInfo.picture,
  796. habook = coreUserInfo.habook,
  797. wechat = coreUserInfo.wechat,
  798. facebook = coreUserInfo.facebook,
  799. google = coreUserInfo.google,
  800. ding = coreUserInfo.ding,
  801. apple = coreUserInfo.apple,
  802. };
  803. return coreUser;
  804. }
  805. else
  806. {
  807. return null;
  808. }
  809. }
  810. else {
  811. return null;
  812. }
  813. }
  814. else if (responseMessage.StatusCode == HttpStatusCode.Unauthorized)
  815. {
  816. return null;
  817. }
  818. else
  819. {
  820. return null;
  821. }
  822. }
  823. }
  824. public class TmdidImplicit
  825. {
  826. public string id_token { get; set; }
  827. public string access_token { get; set; }
  828. public string expires_in { get; set; }
  829. public string token_type { get; set; }
  830. }
  831. public class CoreUser
  832. {
  833. public string searchKey { get; set; }
  834. public string id { get; set; }
  835. public string vid { get; set; }
  836. public string mail { get; set; }
  837. public string mobile { get; set; }
  838. public string name { get; set; }
  839. public string picture { get; set; }
  840. public string habook { get; set; }
  841. public string wechat { get; set; }
  842. public string facebook { get; set; }
  843. public string google { get; set; }
  844. public string ding { get; set; }
  845. public string apple { get; set; }
  846. public bool isActivate { get; set; }
  847. }
  848. public class CoreUserInfo : CoreUser
  849. {
  850. public string area { get; set; }
  851. public string country { get; set; }
  852. public string type { get; set; }
  853. public string password { get; set; }
  854. public string salt { get; set; }
  855. public string passwordOld { get; set; }
  856. public string saltOld { get; set; }
  857. }
  858. }