ScController.cs 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762
  1. using Microsoft.AspNetCore.Mvc;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6. using TEAMModelOS.Models;
  7. using TEAMModelOS.SDK.DI;
  8. using System.Text.Json;
  9. using TEAMModelOS.SDK.Models;
  10. using Microsoft.AspNetCore.Http;
  11. using TEAMModelOS.SDK.Extension;
  12. using Azure.Cosmos;
  13. using System.Text;
  14. using TEAMModelOS.SDK.DI.AzureCosmos.Inner;
  15. using Microsoft.Extensions.Options;
  16. using Azure.Messaging.ServiceBus;
  17. using Microsoft.Extensions.Configuration;
  18. using HTEXLib.COMM.Helpers;
  19. using TEAMModelOS.SDK;
  20. using System.IdentityModel.Tokens.Jwt;
  21. using TEAMModelOS.Services;
  22. using TEAMModelOS.SDK.Models.Service;
  23. using System.IO;
  24. using System.Dynamic;
  25. using Microsoft.AspNetCore.Authorization;
  26. using Azure.Storage.Blobs.Models;
  27. using static TEAMModelOS.SDK.Models.Teacher;
  28. using System.Web;
  29. using static TEAMModelOS.Controllers.FixDataController;
  30. using static TEAMModelOS.SDK.SchoolService;
  31. using Microsoft.AspNetCore.Hosting;
  32. using Microsoft.Azure.Cosmos.Table;
  33. namespace TEAMModelOS.Controllers
  34. {
  35. /// <summary>
  36. ///
  37. /// </summary>
  38. ///
  39. [ProducesResponseType(StatusCodes.Status200OK)]
  40. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  41. //
  42. //[Route("")]
  43. //[Route("api/[controller]")]
  44. [ApiController]
  45. public class ScController : ControllerBase
  46. {
  47. private readonly SnowflakeId _snowflakeId;
  48. private readonly AzureCosmosFactory _azureCosmos;
  49. private readonly DingDing _dingDing;
  50. private readonly Option _option;
  51. private readonly AzureStorageFactory _azureStorage;
  52. private readonly AzureServiceBusFactory _serviceBus;
  53. private readonly AzureRedisFactory _azureRedis;
  54. private readonly CoreAPIHttpService _coreAPIHttpService;
  55. public readonly string type = "scsyxpt";
  56. private readonly HttpTrigger _httpTrigger;
  57. private readonly IWebHostEnvironment _environment;
  58. private readonly ScsStudyApisService _scsStudyApisService;
  59. public IConfiguration _configuration { get; set; }
  60. public ScController(ScsStudyApisService scsStudyApisService, IWebHostEnvironment environment, AzureCosmosFactory azureCosmos, SnowflakeId snowflakeId, DingDing dingDing, IOptionsSnapshot<Option> option, AzureStorageFactory azureStorage,
  61. AzureRedisFactory azureRedis, AzureServiceBusFactory serviceBus, IConfiguration configuration, CoreAPIHttpService coreAPIHttpService, HttpTrigger httpTrigger)
  62. {
  63. _azureCosmos = azureCosmos;
  64. _snowflakeId = snowflakeId;
  65. _dingDing = dingDing;
  66. _option = option?.Value;
  67. _azureStorage = azureStorage;
  68. _serviceBus = serviceBus;
  69. _configuration = configuration;
  70. _azureRedis = azureRedis;
  71. _coreAPIHttpService = coreAPIHttpService;
  72. _httpTrigger = httpTrigger;
  73. _environment = environment;
  74. _scsStudyApisService = scsStudyApisService;
  75. }
  76. /// <summary>
  77. /// 检查教师绑定
  78. /// </summary>
  79. /// <param name="request"></param>
  80. /// <returns></returns>
  81. [ProducesDefaultResponseType]
  82. [HttpPost("sc/check-teacher-bind")]
  83. [AllowAnonymous]
  84. public async Task<IActionResult> CheckTeacherBind(JsonElement request)
  85. {
  86. if (!request.TryGetProperty("schoolId", out JsonElement _schoolId)) return BadRequest();
  87. var table = _azureStorage.GetCloudTableClient().GetTableReference("ScYxpt");
  88. List<ScSchool> schools = await table.FindListByDict<ScSchool>(new Dictionary<string, object>() { { "PartitionKey", "ScSchool" }, { "schoolCode", $"{_schoolId}" } });
  89. if (schools.IsNotEmpty())
  90. {
  91. List<ScTeacher> teachers = await table.FindListByDict<ScTeacher>(new Dictionary<string, object>() { { "PartitionKey", "ScTeacher" }, { "SchoolID", schools[0].schoolid } });
  92. return Ok(new { teachers = teachers });
  93. }
  94. else
  95. {
  96. return Ok();
  97. }
  98. }
  99. /// <summary>
  100. /// 检查醍摩豆id存在多个学校的情况
  101. /// </summary>
  102. /// <param name="request"></param>
  103. /// <returns></returns>
  104. [ProducesDefaultResponseType]
  105. [HttpPost("sc/check-bind")]
  106. [AllowAnonymous]
  107. public async Task<IActionResult> CheckBlobBinds(JsonElement json)
  108. {
  109. List<GroupList> teachers = new List<GroupList>();
  110. string sqs = "select c.members from c where c.pk='GroupList' and c.type='yxtrain' and c.school in ('pjsyzx','pjzx','pjsazx','pjbjxx','pjxnxx','pjthxx','pjcyxx','pjfxxx','pjwjxx','pjzyzx','psywgy','hscjzx','cyhjnz','psasmx','psacjz','pwxjnx','pptsfx','pjjysx','xlzjnx','pdtjnx','pgxjnx','pcjjnx','pdxjnx','pnjyey','pbjyey','pcbyey','pcxyey','pcnyey','xjwhye','pxlyey','sazxye','saxcye','pthyey','psmyey','pshyey','pwxyey','pjysye','pfxyey','pgxyey','pcjyey','pcyhye','pbyyey','pdtyey','xyheye','xhyey','xbeyey','hhzyey','xxyey','saxgye','xllxye','dxxmye','dtxmye','dtydye','pnjdxy','pcxgmy','pcbgqy','saxccq','pjjsjx') ";
  111. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "School").GetItemQueryIterator<GroupList>(queryText: sqs))
  112. {
  113. teachers.Add(item);
  114. }
  115. var a = teachers.SelectMany(x => x.members).GroupBy(y => y.id).ToList();
  116. var ae = a.Select(x => new { key = x.Key, val = x.ToList().Count() });
  117. ae = ae.Where(x => x.val > 1);
  118. return Ok(new { ae });
  119. }
  120. /// <summary>
  121. ///
  122. /// </summary>
  123. /// <param name="request"></param>
  124. /// <returns></returns>
  125. [ProducesDefaultResponseType]
  126. [HttpPost("sc/fix-bind")]
  127. [AllowAnonymous]
  128. public async Task<IActionResult> FixBlobBinds(JsonElement json)
  129. {
  130. List<string> teacherids = new List<string>();
  131. string sql = $" SELECT value(c.id) FROM c where ARRAY_LENGTH(c.binds)>0 ";
  132. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").GetItemQueryIterator<string>(queryText: sql, requestOptions: new QueryRequestOptions() { PartitionKey = new PartitionKey($"Base") }))
  133. {
  134. teacherids.Add(item);
  135. }
  136. var bloblist = await _azureStorage.GetBlobContainerClient("teammodelos").List($"yxpt/scpjx/scbind");
  137. bloblist = bloblist.Select(x => x.Substring(18, 10)).ToList();
  138. List<string> tmdids = teacherids.Except(bloblist).ToList();
  139. List<Teacher> teachers = new List<Teacher>();
  140. sql = $" SELECT value(c) FROM c where ARRAY_LENGTH(c.binds)>0 ";
  141. if (tmdids != null)
  142. {
  143. sql = $"{sql} and c.id in ( {string.Join(",", tmdids.Select(x => $"'{x}'"))} )";
  144. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").GetItemQueryIterator<Teacher>(queryText: sql,
  145. requestOptions: new QueryRequestOptions() { PartitionKey = new PartitionKey($"Base") }))
  146. {
  147. teachers.Add(item);
  148. }
  149. }
  150. List<string> unbind = new List<string>();
  151. List<string> list = new List<string>();
  152. HashSet<string> schoolIds = teachers.Where(z => z.schools.IsNotEmpty()).SelectMany(x => x.schools).Where(m => m.status.Equals("join")).Select(y => y.schoolId).ToHashSet();
  153. List<GroupList> groupLists = new List<GroupList>();
  154. foreach (var schoolid in schoolIds)
  155. {
  156. StringBuilder queryText = new StringBuilder($"SELECT distinct value(c) FROM c where c.type='yxtrain'");
  157. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "School").GetItemQueryIterator<GroupList>(queryText: queryText.ToString(),
  158. requestOptions: new QueryRequestOptions() { PartitionKey = new Azure.Cosmos.PartitionKey($"GroupList-{schoolid}") }))
  159. {
  160. groupLists.Add(item);
  161. }
  162. }
  163. foreach (var teacher in teachers)
  164. {
  165. var a = teacher.binds.SelectMany(y => y.data).ToList().Find(x => !string.IsNullOrWhiteSpace(x));
  166. if (a != null)
  167. {
  168. await _azureStorage.GetBlobContainerClient("teammodelos").UploadFileByContainer(a, $"yxpt/scpjx/scbind", $"{teacher.id}.json");
  169. }
  170. else
  171. {
  172. unbind.Add(teacher.id);
  173. }
  174. if (teacher.schools.IsNotEmpty())
  175. {
  176. foreach (var school in teacher.schools)
  177. {
  178. if (!string.IsNullOrWhiteSpace(school.schoolId))
  179. {
  180. if (school.status.Equals("join"))
  181. {
  182. List<GroupList> yxtrain = groupLists.FindAll(x => x.code.Equals($"GroupList-{school.schoolId}"));
  183. if (yxtrain.IsNotEmpty())
  184. {
  185. var meber = yxtrain.SelectMany(x => x.members).Where(y => y.id.Equals(teacher.id));
  186. //不在研修名单
  187. if (meber == null || meber.Count() <= 0)
  188. {
  189. yxtrain[0].members.Add(new Member { id = teacher.id, type = 1 });
  190. await GroupListService.UpsertList(yxtrain[0], _azureCosmos, _configuration, _serviceBus);
  191. }
  192. }
  193. else
  194. {
  195. GroupList groupList = new GroupList()
  196. {
  197. id = Guid.NewGuid().ToString(),
  198. code = $"GroupList-{school.schoolId}",
  199. creatorId = teacher.id,
  200. type = "yxtrain",
  201. year = DateTimeOffset.UtcNow.Year,
  202. members = new List<Member> { new Member { id = teacher.id, type = 1 } },
  203. scope = "school",
  204. school = school.schoolId,
  205. name = "研修名单",
  206. pk = "GroupList",
  207. ttl = -1,
  208. expire=0,
  209. };
  210. await GroupListService.UpsertList(groupList, _azureCosmos, _configuration, _serviceBus);
  211. }
  212. }
  213. Azure.Response response = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "School").ReadItemStreamAsync(teacher.id, new PartitionKey($"Teacher-{school.schoolId}"));
  214. if (response.Status != 200)
  215. {
  216. SchoolTeacher schoolTeacher = new SchoolTeacher
  217. {
  218. id = teacher.id,
  219. code = $"Teacher-{school.schoolId}",
  220. pk = "Teacher",
  221. name = teacher.name,
  222. picture = teacher.picture,
  223. size = 0,
  224. roles = new List<string> { "teacher" },
  225. permissions = new List<string>(),
  226. status = school.status,
  227. createTime = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds(),
  228. ttl = -1
  229. };
  230. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "School").CreateItemAsync(teacher.id, new PartitionKey($"Teacher-{school.schoolId}"));
  231. }
  232. else
  233. {
  234. JsonDocument document = await JsonDocument.ParseAsync(response.ContentStream);
  235. SchoolTeacher schoolTeacher = document.RootElement.ToObject<SchoolTeacher>();
  236. schoolTeacher.status = school.status;
  237. schoolTeacher.pk = "Teacher";
  238. schoolTeacher.name = teacher.name;
  239. schoolTeacher.picture = teacher.picture;
  240. if (!schoolTeacher.roles.IsEmpty())
  241. {
  242. if (!schoolTeacher.roles.Contains("teacher"))
  243. {
  244. schoolTeacher.roles .Add("teacher");
  245. }
  246. }
  247. else {
  248. schoolTeacher.roles = new List<string> { "teacher" };
  249. }
  250. schoolTeacher.permissions = schoolTeacher.permissions.IsNotEmpty() ? schoolTeacher.permissions : new List<string>();
  251. schoolTeacher.createTime = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds();
  252. schoolTeacher.ttl = -1;
  253. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "School").ReplaceItemAsync(schoolTeacher, schoolTeacher.id, new PartitionKey($"Teacher-{school.schoolId}"));
  254. }
  255. }
  256. }
  257. }
  258. }
  259. return Ok(new { unbind, list });
  260. }
  261. /// <summary>
  262. ///
  263. /// </summary>
  264. /// <param name="request"></param>
  265. /// <returns></returns>
  266. [ProducesDefaultResponseType]
  267. [HttpPost("sc/bind")]
  268. [AllowAnonymous]
  269. public async Task<IActionResult> Bind(SSO sso)
  270. {
  271. try
  272. {
  273. Teacher teacher = null;
  274. TmdidImplicit tmdidImplicit = null;
  275. if (!string.IsNullOrWhiteSpace(sso.mobile))
  276. {
  277. var coreUser = await _coreAPIHttpService.GetUserInfo(new Dictionary<string, string> { { "key", sso.mobile } }, _option.Location, _configuration);
  278. if (coreUser != null)
  279. {
  280. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  281. tmdidImplicit = await _coreAPIHttpService.Implicit(new Dictionary<string, string> { { "grant_type", "implicit" },
  282. { "client_id",clientID },
  283. { "account",coreUser.id },
  284. { "nonce",Guid.NewGuid().ToString()} }, _option.Location, _configuration);
  285. if (tmdidImplicit != null && !string.IsNullOrWhiteSpace(tmdidImplicit.id_token))
  286. {
  287. sso.id_token = tmdidImplicit.id_token;
  288. }
  289. else
  290. {
  291. return Ok(new
  292. {
  293. location = _option.Location,
  294. status = 2,
  295. });
  296. }
  297. }
  298. else {
  299. return Ok(new
  300. {
  301. location = _option.Location,
  302. status = 2,
  303. });
  304. }
  305. }
  306. if (string.IsNullOrWhiteSpace(sso.id_token))
  307. {
  308. return Ok(new
  309. {
  310. location = _option.Location,
  311. status = 2,
  312. });
  313. }
  314. JwtSecurityToken jwt = null;
  315. try
  316. {
  317. jwt = new JwtSecurityToken(sso.id_token);
  318. }
  319. catch (Exception ex)
  320. {
  321. await _dingDing.SendBotMsg($"OS,{_option.Location}\n绑定失败,出现的原因可能是 参数异常:\n{sso.ToJsonString()},{ex.Message}\n{ex.StackTrace}\n", GroupNames.醍摩豆服務運維群組);
  322. return BadRequest();
  323. }
  324. var id = jwt.Payload.Sub;
  325. CoreUser coreUserById= await _coreAPIHttpService.GetUserInfo(new Dictionary<string, string> { { "key", $"{id}" } }, _option.Location, _configuration);
  326. if (coreUserById == null|| string.IsNullOrWhiteSpace(coreUserById.mobile)|| coreUserById.mobile.Length!=11)
  327. {
  328. return Ok(new
  329. {
  330. location = _option.Location,
  331. status = 2,
  332. });
  333. }
  334. jwt.Payload.TryGetValue("name", out object name);
  335. jwt.Payload.TryGetValue("picture", out object picture);
  336. ScSSOData scsso = HttpUtility.UrlDecode(sso.param, Encoding.UTF8).ToObject<ScSSOData>();
  337. var table = _azureStorage.GetCloudTableClient().GetTableReference("ScYxpt");
  338. List<ScTeacher> scTeachers = await table.FindListByDict<ScTeacher>(new Dictionary<string, object> { { Constant.PartitionKey, "ScTeacher" }, { "tmdid", id } });
  339. if (scTeachers.Count > 0 && !string.IsNullOrWhiteSpace(scTeachers[0].RowKey) && !scTeachers[0].RowKey.Equals(scsso.Pxid))
  340. {
  341. return Ok(new
  342. {
  343. location = _option.Location,
  344. //账号已被别的醍摩豆id绑定
  345. status = 3,
  346. tmdid = $"{id}",
  347. name = name.Equals(scTeachers[0].TeacherName) ? $"{name}" : $"{name}({scTeachers[0].TeacherName})",
  348. tid = scsso.tid
  349. });
  350. }
  351. var client = _azureCosmos.GetCosmosClient();
  352. try
  353. {
  354. teacher = await client.GetContainer(Constant.TEAMModelOS, "Teacher").ReadItemAsync<Teacher>(id, new PartitionKey("Base"));
  355. //先检查绑定的平台是否已经被绑定
  356. //四川研训平台跳转隐式登录/或者绑定IES平台接入规范
  357. string sql = $"SELECT distinct value(c) FROM c join A1 in c.binds where A1.tid='{scsso.tid}'";
  358. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").GetItemQueryIterator<Teacher>(queryText: sql,
  359. requestOptions: new QueryRequestOptions() { PartitionKey = new PartitionKey($"Base") }))
  360. {
  361. teacher = item;
  362. break;
  363. }
  364. if (teacher != null)
  365. {
  366. if (teacher.id.Equals(id))
  367. {
  368. var bindData = scsso.data.ToObject<ScBindData>();
  369. //var bind = teacher.binds.Find(x => x.source.Equals($"{scsso.Webid}") && x.userid.Equals($"{scsso.tid}"));
  370. var bind = teacher.binds.Find(x => x.userid.Equals($"{scsso.tid}"));
  371. if (bind == null)
  372. {
  373. teacher.binds = new List<Teacher.ThirdBind> { new Teacher.ThirdBind { data = new List<string> { scsso.data }, userid = $"{scsso.tid}", account = scsso.account, username = scsso.username, type = type } };
  374. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").ReplaceItemAsync<Teacher>(teacher, teacher.id, new PartitionKey(teacher.code));
  375. }
  376. else
  377. {
  378. bind.username = scsso.username;
  379. bind.account = scsso.account;
  380. bool isnew = true;
  381. for (int index = 0; index < bind.data.Count; index++)
  382. {
  383. ScBindData scBind = bind.data[index].ToObject<ScBindData>();
  384. if (scBind.pxid.Equals(bindData.pxid))
  385. {
  386. bind.data[index] = bindData.ToJsonString();
  387. isnew = false;
  388. }
  389. }
  390. if (isnew)
  391. {
  392. bind.data.Add(bindData.ToJsonString());
  393. }
  394. if (bindData != null)
  395. {
  396. bindData.userid = scsso.tid;
  397. bindData.username = scsso.username;
  398. bindData.account = scsso.account;
  399. }
  400. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").ReplaceItemAsync<Teacher>(teacher, teacher.id, new PartitionKey(teacher.code));
  401. }
  402. await _azureStorage.GetBlobContainerClient("teammodelos").UploadFileByContainer(bindData.ToJsonString(), $"yxpt/{scsso.path}/scbind", $"{teacher.id}.json");
  403. await ThirdService.GetScTeacher(bindData, teacher, _azureStorage, _azureCosmos, _serviceBus, _configuration, _dingDing);
  404. }
  405. else
  406. {
  407. return Ok(new
  408. {
  409. location = _option.Location,
  410. //账号已被别的醍摩豆id绑定
  411. status = 3,
  412. tmdid = teacher.id,
  413. name = teacher.name,
  414. tid = scsso.tid
  415. });
  416. }
  417. }
  418. }
  419. catch (CosmosException ex) when (ex.Status ==404)
  420. {
  421. teacher = new Teacher
  422. {
  423. createTime = DateTimeOffset.UtcNow.ToUnixTimeMilliseconds(),
  424. id = id,
  425. pk = "Teacher",
  426. code = "Base",
  427. name = name?.ToString(),
  428. picture = picture?.ToString(),
  429. //创建账号并第一次登录IES5则默认赠送1G
  430. size = 1,
  431. defaultSchool = null,
  432. schools = new List<Teacher.TeacherSchool>(),
  433. binds = new List<Teacher.ThirdBind> { new Teacher.ThirdBind { username = scsso.username, account = scsso.account, data = new List<string> { scsso.data }, userid = $"{scsso.tid}", /*source = $"{scsso.Webid}",*/ type = type } }
  434. };
  435. var container = _azureStorage.GetBlobContainerClient(id);
  436. await container.CreateIfNotExistsAsync(PublicAccessType.None); //嘗試創建Teacher私有容器,如存在則不做任何事,保障容器一定存在
  437. teacher = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").CreateItemAsync<Teacher>(teacher, new PartitionKey("Base"));
  438. ScBindData bindData = scsso.data.ToObject<ScBindData>();
  439. await ThirdService.GetScTeacher(bindData, teacher, _azureStorage, _azureCosmos, _serviceBus, _configuration, _dingDing);
  440. if (bindData != null)
  441. {
  442. bindData.userid = scsso.tid;
  443. bindData.username = scsso.username;
  444. bindData.account = scsso.account;
  445. await _azureStorage.GetBlobContainerClient("teammodelos").UploadFileByContainer(bindData.ToJsonString(), $"yxpt/{scsso.path}/scbind", $"{teacher.id}.json");
  446. }
  447. else
  448. {
  449. await _azureStorage.GetBlobContainerClient("teammodelos").UploadFileByContainer(scsso.data.ToJsonString(), $"yxpt/{scsso.path}/scbind", $"{teacher.id}.json");
  450. }
  451. }
  452. catch (Exception ex)
  453. {
  454. await _dingDing.SendBotMsg($"OS,{_option.Location}\n绑定失败:\n{sso.ToJsonString()},{ex.Message}\n{ex.StackTrace}\n", GroupNames.醍摩豆服務運維群組);
  455. return Ok(new
  456. {
  457. location = _option.Location,
  458. status = 2,
  459. });
  460. }
  461. return Ok(new
  462. {
  463. tmdidImplicit = tmdidImplicit,
  464. location = _option.Location,
  465. status = 200,
  466. });
  467. }
  468. catch (Exception ex)
  469. {
  470. await _dingDing.SendBotMsg($"OS,{_option.Location}\n绑定失败:\n{sso.ToJsonString()},{ex.Message}\n{ex.StackTrace}\n", GroupNames.醍摩豆服務運維群組);
  471. return Ok(new
  472. {
  473. location = _option.Location,
  474. status = 2,
  475. });
  476. }
  477. }
  478. /// <summary>
  479. /// 动态地址路由。"config":"scsyxpt","path":"sc{pjx/jinniu}"
  480. /// </summary>
  481. /// <param name="request"></param>
  482. /// <returns></returns>
  483. [HttpGet("sc/gen-sso")]
  484. [AllowAnonymous]
  485. public async Task<IActionResult> GenSso([FromQuery] ScTchTmd tmd) {
  486. if (tmd != null && !string.IsNullOrWhiteSpace(tmd.tmdid))
  487. {
  488. var table = _azureStorage.GetCloudTableClient().GetTableReference("ScYxpt");
  489. List<ScTeacher> scTeachers = await table.FindListByDict<ScTeacher>(new Dictionary<string, object> { { "PartitionKey", "ScTeacher" }, { "tmdid", $"{tmd.tmdid}" } });
  490. if (scTeachers.Any())
  491. {
  492. string ep = $"Pxid={scTeachers[0].PXID}&Webid=1001&tid={scTeachers[0].TID}&time=1646617519";
  493. string Encrypt = Md5Hash.GetMd5String(ep);
  494. string url = $"https://www.teammodel.cn/sc/sso?{ep}&Encrypt={Encrypt}";
  495. return Ok(new { url });
  496. }
  497. else
  498. {
  499. return Ok("暂无ID的数据");
  500. }
  501. }
  502. else
  503. {
  504. return Ok("参数错误!");
  505. }
  506. }
  507. /// <summary>
  508. /// 动态地址路由。"config":"scsyxpt","path":"sc{pjx/jinniu}"
  509. /// </summary>
  510. /// <param name="request"></param>
  511. /// <returns></returns>
  512. [HttpGet("{path}/sso")]
  513. [AllowAnonymous]
  514. public async Task<IActionResult> Sso([FromQuery] ScSSO scsso, string path)
  515. {
  516. string HostName = HttpContext.GetHostName();
  517. if (!string.IsNullOrWhiteSpace(_option.HostName))
  518. {
  519. HostName = _option.HostName;
  520. }
  521. if (path.Equals("jinniu"))
  522. {
  523. if (!_option.Location.Contains("Dep"))
  524. {
  525. HostName = "jinniu.teammodel.cn";
  526. }
  527. }
  528. if (HostName.Equals("www.teammodel.cn")) {
  529. HostName = "scyx.teammodel.cn";
  530. }
  531. //var rurl = new StringBuilder($"https://{_option.HostName}/sso");
  532. var rurl = new StringBuilder($"https://{HostName}/sso");
  533. try
  534. {
  535. string parmas = $"Pxid={scsso.Pxid}&Webid={scsso.Webid}&tid={scsso.tid}&time={scsso.time}";
  536. if (Md5Hash.GetMd5String(parmas).Equals($"{scsso.Encrypt}"))
  537. {
  538. //四川研训平台跳转隐式登录/或者绑定IES平台接入规范
  539. long ssotime = long.Parse($"{scsso.time}");
  540. long nowtime = DateTimeOffset.UtcNow.ToUnixTimeSeconds();
  541. if (nowtime - ssotime > 60 * 10)//10分钟有效期
  542. {
  543. // return Ok(new { status = 2, msg = "登录超时!" });
  544. }
  545. }
  546. else
  547. {
  548. return Redirect(rurl.Append($"?status=1").ToString());
  549. }
  550. var table = _azureStorage.GetCloudTableClient().GetTableReference("ScYxpt");
  551. string qurey = $"PartitionKey {QueryComparisons.Equal} 'ScTeacher' and TID {QueryComparisons.Equal} {scsso.tid} and RowKey {QueryComparisons.Equal} '{scsso.Pxid}' ";
  552. var result = await table.ExecuteQuerySegmentedAsync(new TableQuery<ScTeacher>().Where(qurey), null);
  553. List<ScTeacher> scTeachers = result.Results;
  554. // List<ScTeacher> scTeachers = await table.FindListByDict<ScTeacher>(new Dictionary<string, object> { { "PartitionKey", "ScTeacher" }, { "TID", scsso.tid }, { "RowKey", $"{scsso.Pxid}" } });
  555. if (!scTeachers.IsNotEmpty())
  556. {
  557. //没有同步省平台
  558. string enurl = $"status=5&param={HttpUtility.UrlEncode(new { scsso.Pxid, scsso.tid }.ToJsonString(), Encoding.UTF8)}";
  559. return Redirect(rurl.Append($"?{enurl}").ToString());
  560. }
  561. string setsql = $"select value(c) from c where c.id ='{scTeachers[0].areaId}' and contains(c.accessConfig,'{ scTeachers[0].ProjectID}') and contains(c.accessConfig,'{ scTeachers[0].ProjectItemID}') and contains(c.accessConfig,'scsyxpt') ";
  562. AreaSetting setting = null;
  563. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Normal").GetItemQueryIterator<AreaSetting>(queryText: setsql,
  564. requestOptions: new QueryRequestOptions() { PartitionKey = new PartitionKey($"AreaSetting") }))
  565. {
  566. setting = item;
  567. break;
  568. }
  569. if (setting == null || (setting != null && string.IsNullOrWhiteSpace(setting.accessConfig)))
  570. {
  571. return Redirect(rurl.Append($"?status=1").ToString());
  572. }
  573. string accessConfig = setting.accessConfig;
  574. Dictionary<string, object> dict = new() { { "accessConfig", accessConfig }, { "pxid", scsso.Pxid }, { "tid", scsso.tid }, { "areaId", $"{setting.id}" } };
  575. string SchoolName = "", SchoolID = "", ProjectID = "", ProjectItemID = "", TeacherName = "", Account = "";
  576. if (scTeachers.IsNotEmpty())
  577. {
  578. ScTeacher scTeacher = scTeachers[0];
  579. if (scTeacher != null && $"{scTeacher.PXID}".Equals(scsso.Pxid) && $"{scTeacher.TID}".Equals(scsso.tid))
  580. {
  581. SchoolName = scTeacher.SchoolName;
  582. SchoolID = $"{scTeacher.SchoolID}";
  583. ProjectID = $"{ scTeacher.ProjectID}";
  584. ProjectItemID = $"{ scTeacher.ProjectItemID}";
  585. TeacherName = $"{ scTeacher.TeacherName}";
  586. Account = $"{ scTeacher.Account}";
  587. }
  588. }
  589. if (string.IsNullOrWhiteSpace(SchoolID))
  590. {
  591. //(int status, string json) = await _httpTrigger.RequestHttpTrigger(dict, _option.Location, "GetSingleTeacherByProject");
  592. (int status, string json) = await _scsStudyApisService.GetSingleTeacherByProject(setting.id, accessConfig, scsso.Pxid, scsso.tid);
  593. if (status == 200)
  594. {
  595. ScTeacher scTeacher = json.ToObject<ScTeacher>(new JsonSerializerOptions { PropertyNameCaseInsensitive = false });
  596. if (scTeacher != null && $"{scTeacher.PXID}".Equals(scsso.Pxid) && $"{scTeacher.TID}".Equals(scsso.tid))
  597. {
  598. SchoolName = scTeacher.SchoolName;
  599. SchoolID = $"{scTeacher.SchoolID}";
  600. ProjectID = $"{ scTeacher.ProjectID}";
  601. ProjectItemID = $"{ scTeacher.ProjectItemID}";
  602. TeacherName = $"{ scTeacher.TeacherName}";
  603. Account = $"{ scTeacher.Account}";
  604. }
  605. }
  606. else
  607. {
  608. await _dingDing.SendBotMsg($"OS,{_option.Location}\n省平台教师信息:\nstatus:{status}{json}\n{dict.ToJsonString()} \nGetSingleTeacherByProject", GroupNames.醍摩豆服務運維群組);
  609. }
  610. }
  611. ScBindData bindData = new()
  612. {
  613. sn = SchoolName,
  614. sid = SchoolID,
  615. pd = ProjectID,
  616. pid = ProjectItemID,
  617. pxid = scsso.Pxid,
  618. userid = scsso.tid,
  619. username = TeacherName,
  620. account = Account,
  621. path = path,
  622. };
  623. var data = bindData.ToJsonString();
  624. ScSSOData sso = new ScSSOData
  625. {
  626. username = TeacherName,
  627. account = Account,
  628. path = path,
  629. Pxid = scsso.Pxid,
  630. Encrypt = scsso.Encrypt,
  631. tid = scsso.tid,
  632. time = scsso.time,
  633. data = data
  634. };
  635. Teacher teacher = null;
  636. //四川研训平台跳转隐式登录/或者绑定IES平台接入规范
  637. //string sql = $"SELECT distinct value(c) FROM c join A1 in c.binds where A1.pxid='{sso.Pxid}' and A1.webid='{sso.Webid}' and A1.tid='{sso.tid}'";
  638. string sql = $"SELECT distinct value(c) FROM c join A1 in c.binds where A1.userid='{sso.tid}'";
  639. await foreach (var item in _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").GetItemQueryIterator<Teacher>(queryText: sql,
  640. requestOptions: new QueryRequestOptions() { PartitionKey = new PartitionKey($"Base") }))
  641. {
  642. teacher = item;
  643. break;
  644. }
  645. if (teacher == null)
  646. {
  647. //string enurl = HttpUtility.UrlEncode(rurl.Append($"?status=4&param={sso.ToJsonString()}&type={type}&bindurl=sc/bind").ToString());
  648. string enurl = $"status=4&param={HttpUtility.UrlEncode(sso.ToJsonString(), Encoding.UTF8)}&type={type}&bindurl=sc/bind";
  649. return Redirect(rurl.Append($"?{enurl}").ToString());
  650. }
  651. else
  652. {
  653. var clientID = _configuration.GetValue<string>("HaBookAuth:CoreService:clientID");
  654. var location = _option.Location;
  655. TmdidImplicit implicit_token = await _coreAPIHttpService.Implicit(
  656. new Dictionary<string, string>()
  657. {
  658. { "grant_type", "implicit" },
  659. { "client_id",clientID },
  660. { "account",teacher.id },
  661. { "nonce",Guid.NewGuid().ToString()}
  662. }, location, _configuration);
  663. if (implicit_token != null)
  664. {
  665. if (string.IsNullOrWhiteSpace(implicit_token.id_token))
  666. {
  667. await _dingDing.SendBotMsg($"OS,隐式登录获得信息为空:{_option.Location}-\n{scsso.ToJsonString()} \npath:{path}\n{implicit_token.ToJsonString()}", GroupNames.醍摩豆服務運維群組);
  668. return Redirect(rurl.Append($"?status=1").ToString());
  669. }
  670. //处理自动加入学校,加入培训名单
  671. await ThirdService.GetScTeacher(bindData, teacher, _azureStorage, _azureCosmos, _serviceBus, _configuration, _dingDing);
  672. var bind = teacher.binds.Find(x => x.userid.Equals(sso.tid));
  673. //var bind = teacher.binds.Find(x => x.userid.Equals(sso.tid) && x.source.Equals(sso.Webid));
  674. if (bind != null)
  675. {
  676. bool isnew = true;
  677. for (int index = 0; index < bind.data.Count; index++)
  678. {
  679. ScBindData scBind = bind.data[index].ToObject<ScBindData>();
  680. if (scBind.pxid.Equals(bindData.pxid))
  681. {
  682. bind.data[index] = bindData.ToJsonString();
  683. isnew = false;
  684. }
  685. }
  686. if (isnew)
  687. {
  688. bind.data.Add(bindData.ToJsonString());
  689. }
  690. bind.username = TeacherName;
  691. bind.account = Account;
  692. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").ReplaceItemAsync<Teacher>(teacher, teacher.id, new PartitionKey(teacher.code));
  693. }
  694. try
  695. {
  696. await _azureStorage.GetBlobContainerClient("teammodelos").UploadFileByContainer(bindData.ToJsonString(), $"yxpt/{sso.path}/scbind", $"{teacher.id}.json");
  697. }
  698. catch (Exception ex)
  699. {
  700. await _dingDing.SendBotMsg($"OS,{_option.Location}-\n文件失败 \npath:{path}\n\n{ex.Message}\n{ex.StackTrace}\n yxpt/{sso.path}/scbind/{teacher.id}.json", GroupNames.醍摩豆服務運維群組);
  701. }
  702. rurl.Append($"?status=200&id_token={implicit_token.id_token}&access_token={implicit_token.access_token}&expires_in={HttpUtility.UrlEncode(implicit_token.expires_in)}&token_type={HttpUtility.UrlEncode(implicit_token.token_type)}").ToString();
  703. string uri = rurl.ToString();
  704. return Redirect(uri);
  705. }
  706. else
  707. {
  708. //绑定失效
  709. //if (teacher.binds.IsNotEmpty())
  710. //{
  711. // teacher.binds.RemoveAll(x => x.userid.Equals(sso.tid));
  712. // await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, "Teacher").ReplaceItemAsync<Teacher>(teacher, teacher.id, new PartitionKey(teacher.code));
  713. //}
  714. //string enurl = $"status=4&param={HttpUtility.UrlEncode(sso.ToJsonString(), Encoding.UTF8)}&type={type}&bindurl=sc/bind";
  715. // return Redirect(rurl.Append($"?{enurl}").ToString());
  716. return Redirect(rurl.Append($"?status=1").ToString());
  717. }
  718. }
  719. }
  720. catch (Exception ex)
  721. {
  722. await _dingDing.SendBotMsg($"OS,{_option.Location}-\n{scsso.ToJsonString()} \npath:{path}\n\n{ex.Message}\n{ex.StackTrace}", GroupNames.醍摩豆服務運維群組);
  723. return Redirect(rurl.Append($"?status=1").ToString());
  724. }
  725. }
  726. public class DbBind
  727. {
  728. public string id { get; set; }
  729. public string userid { get; set; }
  730. public List<string> data { get; set; }
  731. }
  732. }
  733. }