SysLdapService.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using Novell.Directory.Ldap;
  7. namespace Admin.NET.Core;
  8. /// <summary>
  9. /// 系统域登录配置服务 🧩
  10. /// </summary>
  11. [ApiDescriptionSettings(Order = 496)]
  12. public class SysLdapService : IDynamicApiController, ITransient
  13. {
  14. private SqlSugarRepository<SysLdap> sysLdapRep = null;
  15. public SysLdapService()
  16. {
  17. }
  18. public SqlSugarRepository<SysLdap> SysLdapRep
  19. {
  20. get
  21. {
  22. sysLdapRep ??= App.GetRequiredService<SqlSugarRepository<SysLdap>>();
  23. return sysLdapRep;
  24. }
  25. }
  26. /// <summary>
  27. /// 获取系统域登录配置分页列表 🔖
  28. /// </summary>
  29. /// <param name="input"></param>
  30. /// <returns></returns>
  31. [DisplayName("获取系统域登录配置分页列表")]
  32. public async Task<SqlSugarPagedList<SysLdap>> Page(SysLdapInput input)
  33. {
  34. return await SysLdapRep.AsQueryable()
  35. .WhereIF(!string.IsNullOrWhiteSpace(input.SearchKey), u => u.Host.Contains(input.SearchKey.Trim()))
  36. .WhereIF(!string.IsNullOrWhiteSpace(input.Host), u => u.Host.Contains(input.Host.Trim()))
  37. .OrderBy(u => u.CreateTime, OrderByType.Desc)
  38. .ToPagedListAsync(input.Page, input.PageSize);
  39. }
  40. /// <summary>
  41. /// 增加系统域登录配置 🔖
  42. /// </summary>
  43. /// <param name="input"></param>
  44. /// <returns></returns>
  45. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  46. [DisplayName("增加系统域登录配置")]
  47. public async Task<long> Add(AddSysLdapInput input)
  48. {
  49. var entity = input.Adapt<SysLdap>();
  50. entity.BindPass = CryptogramUtil.Encrypt(input.BindPass);
  51. await SysLdapRep.InsertAsync(entity);
  52. return entity.Id;
  53. }
  54. /// <summary>
  55. /// 更新系统域登录配置 🔖
  56. /// </summary>
  57. /// <param name="input"></param>
  58. /// <returns></returns>
  59. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  60. [DisplayName("更新系统域登录配置")]
  61. public async Task Update(UpdateSysLdapInput input)
  62. {
  63. var entity = input.Adapt<SysLdap>();
  64. if (!string.IsNullOrEmpty(input.BindPass) && input.BindPass.Length < 32)
  65. {
  66. entity.BindPass = CryptogramUtil.Encrypt(input.BindPass); // 加密
  67. }
  68. await SysLdapRep.AsUpdateable(entity).IgnoreColumns(ignoreAllNullColumns: true).ExecuteCommandAsync();
  69. }
  70. /// <summary>
  71. /// 删除系统域登录配置 🔖
  72. /// </summary>
  73. /// <param name="input"></param>
  74. /// <returns></returns>
  75. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  76. [DisplayName("删除系统域登录配置")]
  77. public async Task Delete(DeleteSysLdapInput input)
  78. {
  79. var entity = await SysLdapRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  80. await SysLdapRep.FakeDeleteAsync(entity); // 假删除
  81. //await _rep.DeleteAsync(entity); // 真删除
  82. }
  83. /// <summary>
  84. /// 获取系统域登录配置详情 🔖
  85. /// </summary>
  86. /// <param name="input"></param>
  87. /// <returns></returns>
  88. [DisplayName("获取系统域登录配置详情")]
  89. public async Task<SysLdap> GetDetail([FromQuery] DetailSysLdapInput input)
  90. {
  91. return await SysLdapRep.GetFirstAsync(u => u.Id == input.Id);
  92. }
  93. /// <summary>
  94. /// 获取系统域登录配置列表 🔖
  95. /// </summary>
  96. /// <returns></returns>
  97. [DisplayName("获取系统域登录配置列表")]
  98. public async Task<List<SysLdap>> GetList()
  99. {
  100. return await SysLdapRep.AsQueryable().Select<SysLdap>().ToListAsync();
  101. }
  102. /// <summary>
  103. /// 验证账号
  104. /// </summary>
  105. /// <param name="account">域用户</param>
  106. /// <param name="password">密码</param>
  107. /// <param name="tenantId">租户</param>
  108. /// <returns></returns>
  109. [NonAction]
  110. public async Task<bool> AuthAccount(long tenantId, string account, string password)
  111. {
  112. var sysLdap = await SysLdapRep.GetFirstAsync(u => u.TenantId == tenantId) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  113. var ldapConn = new LdapConnection();
  114. try
  115. {
  116. ldapConn.Connect(sysLdap.Host, sysLdap.Port);
  117. ldapConn.Bind(sysLdap.Version, sysLdap.BindDn, sysLdap.BindPass);
  118. var ldapSearchResults = ldapConn.Search(sysLdap.BaseDn, LdapConnection.ScopeSub, sysLdap.AuthFilter.Replace("$s", account), null, false);
  119. string dn = string.Empty;
  120. while (ldapSearchResults.HasMore())
  121. {
  122. var ldapEntry = ldapSearchResults.Next();
  123. var sAMAccountName = ldapEntry.GetAttribute(sysLdap.AuthFilter)?.StringValue;
  124. if (!string.IsNullOrEmpty(sAMAccountName))
  125. {
  126. dn = ldapEntry.Dn;
  127. break;
  128. }
  129. }
  130. if (string.IsNullOrEmpty(dn)) throw Oops.Oh(ErrorCodeEnum.D1002);
  131. // var attr = new LdapAttribute("userPassword", password);
  132. ldapConn.Bind(dn, password);
  133. }
  134. catch (LdapException e)
  135. {
  136. return e.ResultCode switch
  137. {
  138. LdapException.NoSuchObject or LdapException.NoSuchAttribute => throw Oops.Oh(ErrorCodeEnum.D0009),
  139. LdapException.InvalidCredentials => false,
  140. _ => throw Oops.Oh(e.Message),
  141. };
  142. }
  143. finally
  144. {
  145. ldapConn.Disconnect();
  146. }
  147. return true;
  148. }
  149. /// <summary>
  150. /// 同步域用户 🔖
  151. /// </summary>
  152. /// <param name="input"></param>
  153. /// <returns></returns>
  154. [DisplayName("同步域用户")]
  155. public async Task SyncUser(SyncSysLdapInput input)
  156. {
  157. var sysLdap = await SysLdapRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  158. var ldapConn = new LdapConnection();
  159. try
  160. {
  161. ldapConn.Connect(sysLdap.Host, sysLdap.Port);
  162. ldapConn.Bind(sysLdap.Version, sysLdap.BindDn, sysLdap.BindPass);
  163. var ldapSearchResults = ldapConn.Search(sysLdap.BaseDn, LdapConnection.ScopeOne, "(objectClass=*)", null, false);
  164. var userLdapList = new List<SysUserLdap>();
  165. while (ldapSearchResults.HasMore())
  166. {
  167. LdapEntry ldapEntry;
  168. try
  169. {
  170. ldapEntry = ldapSearchResults.Next();
  171. if (ldapEntry == null) continue;
  172. }
  173. catch (LdapException)
  174. {
  175. continue;
  176. }
  177. var attrs = ldapEntry.GetAttributeSet();
  178. var deptCode = GetDepartmentCode(attrs, sysLdap.BindAttrCode);
  179. if (attrs.Count == 0 || attrs.ContainsKey("OU"))
  180. {
  181. SearchDnLdapUser(ldapConn, sysLdap, userLdapList, ldapEntry.Dn, deptCode);
  182. }
  183. else
  184. {
  185. var sysUserLdap = CreateSysUserLdap(attrs, sysLdap.BindAttrAccount, sysLdap.BindAttrEmployeeId, deptCode);
  186. if (string.IsNullOrEmpty(sysUserLdap.EmployeeId)) continue;
  187. userLdapList.Add(sysUserLdap);
  188. }
  189. }
  190. if (userLdapList.Count == 0)
  191. return;
  192. await App.GetRequiredService<SysUserLdapService>().InsertUserLdaps(sysLdap.TenantId!.Value, userLdapList);
  193. }
  194. catch (LdapException e)
  195. {
  196. throw e.ResultCode switch
  197. {
  198. LdapException.NoSuchObject or LdapException.NoSuchAttribute => Oops.Oh(ErrorCodeEnum.D0009),
  199. _ => Oops.Oh(e.Message),
  200. };
  201. }
  202. finally
  203. {
  204. ldapConn.Disconnect();
  205. }
  206. }
  207. /// <summary>
  208. /// 获取部门代码
  209. /// </summary>
  210. /// <param name="attrs"></param>
  211. /// <param name="bindAttrCode"></param>
  212. /// <returns></returns>
  213. private static string GetDepartmentCode(LdapAttributeSet attrs, string bindAttrCode)
  214. {
  215. return bindAttrCode == "objectGUID"
  216. ? new Guid(attrs.GetAttribute(bindAttrCode)?.ByteValue).ToString()
  217. : attrs.GetAttribute(bindAttrCode)?.StringValue ?? "0";
  218. }
  219. /// <summary>
  220. /// 创建同步对象
  221. /// </summary>
  222. /// <param name="attrs"></param>
  223. /// <param name="bindAttrAccount"></param>
  224. /// <param name="bindAttrEmployeeId"></param>
  225. /// <param name="deptCode"></param>
  226. /// <returns></returns>
  227. private static SysUserLdap CreateSysUserLdap(LdapAttributeSet attrs, string bindAttrAccount, string bindAttrEmployeeId, string deptCode)
  228. {
  229. return new SysUserLdap
  230. {
  231. Account = !attrs.ContainsKey(bindAttrAccount) ? null : attrs.GetAttribute(bindAttrAccount)?.StringValue,
  232. EmployeeId = !attrs.ContainsKey(bindAttrEmployeeId) ? null : attrs.GetAttribute(bindAttrEmployeeId)?.StringValue,
  233. DeptCode = deptCode
  234. };
  235. }
  236. /// <summary>
  237. /// 遍历查询域用户
  238. /// </summary>
  239. /// <param name="ldapConn"></param>
  240. /// <param name="sysLdap"></param>
  241. /// <param name="userLdapList"></param>
  242. /// <param name="baseDn"></param>
  243. /// <param name="deptCode"></param>
  244. private static void SearchDnLdapUser(LdapConnection ldapConn, SysLdap sysLdap, List<SysUserLdap> userLdapList, string baseDn, string deptCode)
  245. {
  246. var ldapSearchResults = ldapConn.Search(baseDn, LdapConnection.ScopeOne, "(objectClass=*)", null, false);
  247. while (ldapSearchResults.HasMore())
  248. {
  249. LdapEntry ldapEntry;
  250. try
  251. {
  252. ldapEntry = ldapSearchResults.Next();
  253. if (ldapEntry == null) continue;
  254. }
  255. catch (LdapException)
  256. {
  257. continue;
  258. }
  259. var attrs = ldapEntry.GetAttributeSet();
  260. deptCode = GetDepartmentCode(attrs, sysLdap.BindAttrCode);
  261. if (attrs.Count == 0 || attrs.ContainsKey("OU"))
  262. SearchDnLdapUser(ldapConn, sysLdap, userLdapList, ldapEntry.Dn, deptCode);
  263. else
  264. {
  265. var sysUserLdap = CreateSysUserLdap(attrs, sysLdap.BindAttrAccount, sysLdap.BindAttrEmployeeId, deptCode);
  266. if (string.IsNullOrEmpty(sysUserLdap.EmployeeId)) continue;
  267. userLdapList.Add(sysUserLdap);
  268. }
  269. }
  270. }
  271. /// <summary>
  272. /// 同步域组织 🔖
  273. /// </summary>
  274. /// <param name="input"></param>
  275. /// <returns></returns>
  276. [DisplayName("同步域组织")]
  277. public async Task SyncDept(SyncSysLdapInput input)
  278. {
  279. var sysLdap = await SysLdapRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  280. var ldapConn = new LdapConnection();
  281. try
  282. {
  283. ldapConn.Connect(sysLdap.Host, sysLdap.Port);
  284. ldapConn.Bind(sysLdap.Version, sysLdap.BindDn, sysLdap.BindPass);
  285. var ldapSearchResults = ldapConn.Search(sysLdap.BaseDn, LdapConnection.ScopeOne, "(objectClass=*)", null, false);
  286. var orgList = new List<SysOrg>();
  287. while (ldapSearchResults.HasMore())
  288. {
  289. LdapEntry ldapEntry;
  290. try
  291. {
  292. ldapEntry = ldapSearchResults.Next();
  293. if (ldapEntry == null) continue;
  294. }
  295. catch (LdapException)
  296. {
  297. continue;
  298. }
  299. var attrs = ldapEntry.GetAttributeSet();
  300. if (attrs.Count == 0 || attrs.ContainsKey("OU"))
  301. {
  302. var sysOrg = CreateSysOrg(attrs, sysLdap, orgList, new SysOrg { Id = 0, Level = 0 });
  303. orgList.Add(sysOrg);
  304. SearchDnLdapDept(ldapConn, sysLdap, orgList, ldapEntry.Dn, sysOrg);
  305. }
  306. }
  307. if (orgList.Count == 0)
  308. return;
  309. await App.GetRequiredService<SysOrgService>().BatchAddOrgs(orgList);
  310. }
  311. catch (LdapException e)
  312. {
  313. throw e.ResultCode switch
  314. {
  315. LdapException.NoSuchObject or LdapException.NoSuchAttribute => Oops.Oh(ErrorCodeEnum.D0009),
  316. _ => Oops.Oh(e.Message),
  317. };
  318. }
  319. finally
  320. {
  321. ldapConn.Disconnect();
  322. }
  323. }
  324. /// <summary>
  325. /// 遍历查询域用户
  326. /// </summary>
  327. /// <param name="ldapConn"></param>
  328. /// <param name="sysLdap"></param>
  329. /// <param name="listOrgs"></param>
  330. /// <param name="baseDn"></param>
  331. /// <param name="org"></param>
  332. private static void SearchDnLdapDept(LdapConnection ldapConn, SysLdap sysLdap, List<SysOrg> listOrgs, string baseDn, SysOrg org)
  333. {
  334. var ldapSearchResults = ldapConn.Search(baseDn, LdapConnection.ScopeOne, "(objectClass=*)", null, false);
  335. while (ldapSearchResults.HasMore())
  336. {
  337. LdapEntry ldapEntry;
  338. try
  339. {
  340. ldapEntry = ldapSearchResults.Next();
  341. if (ldapEntry == null) continue;
  342. }
  343. catch (LdapException)
  344. {
  345. continue;
  346. }
  347. var attrs = ldapEntry.GetAttributeSet();
  348. if (attrs.Count == 0 || attrs.ContainsKey("OU"))
  349. {
  350. var sysOrg = CreateSysOrg(attrs, sysLdap, listOrgs, org);
  351. listOrgs.Add(sysOrg);
  352. SearchDnLdapDept(ldapConn, sysLdap, listOrgs, ldapEntry.Dn, sysOrg);
  353. }
  354. }
  355. }
  356. /// <summary>
  357. /// 创建架构对象
  358. /// </summary>
  359. /// <param name="attrs"></param>
  360. /// <param name="sysLdap"></param>
  361. /// <param name="listOrgs"></param>
  362. /// <param name="org"></param>
  363. /// <returns></returns>
  364. private static SysOrg CreateSysOrg(LdapAttributeSet attrs, SysLdap sysLdap, List<SysOrg> listOrgs, SysOrg org)
  365. {
  366. return new SysOrg
  367. {
  368. Pid = org.Id,
  369. Id = YitIdHelper.NextId(),
  370. Code = !attrs.ContainsKey(sysLdap.BindAttrCode) ? null : new Guid(attrs.GetAttribute(sysLdap.BindAttrCode)?.ByteValue).ToString(),
  371. Level = org.Level + 1,
  372. Name = !attrs.ContainsKey(sysLdap.BindAttrAccount) ? null : attrs.GetAttribute(sysLdap.BindAttrAccount)?.StringValue,
  373. OrderNo = listOrgs.Count + 1,
  374. };
  375. }
  376. }