SysOAuthService.cs 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. // 大名科技(天津)有限公司 版权所有
  2. //
  3. // 此源代码遵循位于源代码树根目录中的 LICENSE 文件的许可证
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动
  6. //
  7. // 任何基于本项目二次开发而产生的一切法律纠纷和责任,均与作者无关
  8. using Microsoft.AspNetCore.Authentication;
  9. using System.Security.Claims;
  10. namespace Admin.NET.Core.Service;
  11. /// <summary>
  12. /// 系统OAuth服务 💥
  13. /// </summary>
  14. [AllowAnonymous]
  15. [ApiDescriptionSettings(Order = 495)]
  16. public class SysOAuthService : IDynamicApiController, ITransient
  17. {
  18. private readonly IHttpContextAccessor _httpContextAccessor;
  19. private readonly SqlSugarRepository<SysWechatUser> _sysWechatUserRep;
  20. public SysOAuthService(IHttpContextAccessor httpContextAccessor,
  21. SqlSugarRepository<SysWechatUser> sysWechatUserRep)
  22. {
  23. _httpContextAccessor = httpContextAccessor;
  24. _sysWechatUserRep = sysWechatUserRep;
  25. }
  26. /// <summary>
  27. /// 第三方登录 🔖
  28. /// </summary>
  29. /// <param name="provider"></param>
  30. /// <param name="redirectUrl"></param>
  31. /// <returns></returns>
  32. [ApiDescriptionSettings(Name = "SignIn"), HttpGet]
  33. [DisplayName("第三方登录")]
  34. public virtual async Task<IActionResult> SignIn([FromQuery] string provider, [FromQuery] string redirectUrl)
  35. {
  36. if (string.IsNullOrWhiteSpace(provider) || !await _httpContextAccessor.HttpContext.IsProviderSupportedAsync(provider))
  37. throw Oops.Oh("不支持的OAuth类型");
  38. var request = _httpContextAccessor.HttpContext.Request;
  39. var url = $"{request.Scheme}://{request.Host}{request.PathBase}{request.Path}Callback?provider={provider}&redirectUrl={redirectUrl}";
  40. var properties = new AuthenticationProperties { RedirectUri = url };
  41. properties.Items["LoginProvider"] = provider;
  42. return await Task.FromResult(new ChallengeResult(provider, properties));
  43. }
  44. /// <summary>
  45. /// 授权回调 🔖
  46. /// </summary>
  47. /// <param name="provider"></param>
  48. /// <param name="redirectUrl"></param>
  49. /// <returns></returns>
  50. [ApiDescriptionSettings(Name = "SignInCallback"), HttpGet]
  51. [DisplayName("授权回调")]
  52. public virtual async Task<IActionResult> SignInCallback([FromQuery] string provider = null, [FromQuery] string redirectUrl = "")
  53. {
  54. if (string.IsNullOrWhiteSpace(provider) || !await _httpContextAccessor.HttpContext.IsProviderSupportedAsync(provider))
  55. throw Oops.Oh("不支持的OAuth类型");
  56. var authenticateResult = await _httpContextAccessor.HttpContext.AuthenticateAsync(provider);
  57. if (!authenticateResult.Succeeded)
  58. throw Oops.Oh("授权失败");
  59. var openIdClaim = authenticateResult.Principal.FindFirst(ClaimTypes.NameIdentifier);
  60. if (openIdClaim == null || string.IsNullOrWhiteSpace(openIdClaim.Value))
  61. throw Oops.Oh("授权失败");
  62. var name = authenticateResult.Principal.FindFirst(ClaimTypes.Name)?.Value;
  63. var email = authenticateResult.Principal.FindFirst(ClaimTypes.Email)?.Value;
  64. var mobilePhone = authenticateResult.Principal.FindFirst(ClaimTypes.MobilePhone)?.Value;
  65. var dateOfBirth = authenticateResult.Principal.FindFirst(ClaimTypes.DateOfBirth)?.Value;
  66. var gender = authenticateResult.Principal.FindFirst(ClaimTypes.Gender)?.Value;
  67. var avatarUrl = "";
  68. var platformType = PlatformTypeEnum.微信公众号;
  69. if (provider == "Gitee")
  70. {
  71. platformType = PlatformTypeEnum.Gitee;
  72. avatarUrl = authenticateResult.Principal.FindFirst(OAuthClaim.GiteeAvatarUrl)?.Value;
  73. }
  74. // 若账号不存在则新建
  75. var wechatUser = await _sysWechatUserRep.AsQueryable().Includes(u => u.SysUser).ClearFilter().FirstAsync(u => u.OpenId == openIdClaim.Value);
  76. if (wechatUser == null)
  77. {
  78. var userId = await App.GetRequiredService<SysUserService>().AddUser(new AddUserInput()
  79. {
  80. Account = name,
  81. RealName = name,
  82. NickName = name,
  83. Email = email,
  84. Avatar = avatarUrl,
  85. Phone = mobilePhone,
  86. OrgId = 1300000000101, // 根组织架构
  87. RoleIdList = new List<long> { 1300000000104 } // 仅本人数据角色
  88. });
  89. await _sysWechatUserRep.InsertAsync(new SysWechatUser()
  90. {
  91. UserId = userId,
  92. OpenId = openIdClaim.Value,
  93. Avatar = avatarUrl,
  94. NickName = name,
  95. PlatformType = platformType
  96. });
  97. wechatUser = await _sysWechatUserRep.AsQueryable().Includes(u => u.SysUser).ClearFilter().FirstAsync(u => u.OpenId == openIdClaim.Value);
  98. }
  99. // 构建Token令牌
  100. var token = await App.GetRequiredService<SysAuthService>().CreateToken(wechatUser.SysUser);
  101. return new RedirectResult($"{redirectUrl}/#/login?token={token.AccessToken}");
  102. }
  103. }