AlipayService.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using Admin.NET.Core.Service.Alipay;
  7. using Aop.Api;
  8. using Aop.Api.Domain;
  9. using Aop.Api.Request;
  10. using Aop.Api.Response;
  11. using Aop.Api.Util;
  12. using Microsoft.AspNetCore.Hosting;
  13. // ReSharper disable All
  14. namespace Admin.NET.Core.Service;
  15. /// <summary>
  16. /// 系统登录授权服务 🧩
  17. /// </summary>
  18. [ApiDescriptionSettings(Order = 500)]
  19. public class AlipayService : IDynamicApiController, ITransient
  20. {
  21. private readonly IWebHostEnvironment _webHostEnvironment;
  22. private readonly SysConfigService _sysConfigService;
  23. private readonly IHttpContextAccessor _httpContext;
  24. private readonly AlipayOptions _alipayOptions;
  25. private readonly IAopClient _alipayClient;
  26. private readonly UserManager _userManager;
  27. public AlipayService(
  28. UserManager userManager,
  29. SysConfigService sysConfigService,
  30. IHttpContextAccessor httpContext,
  31. IWebHostEnvironment webHostEnvironment,
  32. IOptions<AlipayOptions> alipayOptions)
  33. {
  34. _userManager = userManager;
  35. _httpContext = httpContext;
  36. _sysConfigService = sysConfigService;
  37. _alipayOptions = alipayOptions.Value;
  38. _webHostEnvironment = webHostEnvironment;
  39. // 初始化支付宝客户端
  40. string path = App.WebHostEnvironment.ContentRootPath;
  41. _alipayClient = new DefaultAopClient(new AlipayConfig
  42. {
  43. Format = "json",
  44. Charset = "UTF-8",
  45. AppId = _alipayOptions.AppId,
  46. SignType = _alipayOptions.SignType,
  47. ServerUrl = _alipayOptions.ServerUrl,
  48. PrivateKey = _alipayOptions.PrivateKey,
  49. EncryptKey = _alipayOptions.EncryptKey,
  50. AppCertPath = Path.Combine(path, _alipayOptions.AppCertPath),
  51. RootCertPath = Path.Combine(path, _alipayOptions.RootCertPath),
  52. AlipayPublicCertPath = Path.Combine(path, _alipayOptions.AlipayPublicCertPath)
  53. });
  54. }
  55. /// <summary>
  56. /// 获取授权信息
  57. /// </summary>
  58. /// <param name="input"></param>
  59. /// <returns></returns>
  60. [NonUnify]
  61. [AllowAnonymous]
  62. [DisplayName("获取授权信息")]
  63. [ApiDescriptionSettings(Name = "GetAuthInfo"), HttpGet]
  64. public ActionResult GetAuthInfo([FromQuery] AlipayAuthInfoInput input)
  65. {
  66. var type = input.UserId.Split('-').FirstOrDefault().ToInt();
  67. var userId = input.UserId.Split('-').LastOrDefault().ToLong();
  68. // 当前网页接口地址
  69. var currentUrl = $"{_httpContext.HttpContext!.Request.GetOrigin()}{_httpContext.HttpContext!.Request.Path}?userId={input.UserId}";
  70. if (string.IsNullOrEmpty(input.AuthCode))
  71. {
  72. // 重新授权
  73. var url = $"{_alipayOptions.AuthUrl}?app_id={_alipayOptions.AppId}&scope=auth_user&redirect_uri={currentUrl}";
  74. return new RedirectResult(url);
  75. }
  76. // 组装授权请求参数
  77. AlipaySystemOauthTokenRequest request = new()
  78. {
  79. GrantType = AlipayConst.GrantType,
  80. Code = input.AuthCode
  81. };
  82. AlipaySystemOauthTokenResponse response = _alipayClient.CertificateExecute(request);
  83. // token换取用户信息
  84. AlipayUserInfoShareRequest infoShareRequest = new();
  85. AlipayUserInfoShareResponse info = _alipayClient.CertificateExecute(infoShareRequest, response.AccessToken);
  86. // 循环执行扫码后需要执行的业务逻辑,需要至少一个继承方法返回true,否则抛出异常
  87. var pass = false;
  88. foreach (var notify in App.GetServices<IAlipayNotify>())
  89. if (notify.ScanCallback(type, userId, info)) pass = true;
  90. if (!pass) throw Oops.Oh("未处理的授权逻辑");
  91. // 执行完,重定向到指定界面
  92. var authPageUrl = _sysConfigService.GetConfigValue<string>(ConfigConst.AlipayAuthPageUrl + type).Result;
  93. return new RedirectResult(authPageUrl);
  94. }
  95. /// <summary>
  96. /// 支付回调
  97. /// </summary>
  98. /// <returns></returns>
  99. [AllowAnonymous]
  100. [DisplayName("支付回调")]
  101. [ApiDescriptionSettings(Name = "Notify"), HttpPost]
  102. public string Notify()
  103. {
  104. SortedDictionary<string, string> sorted = new();
  105. foreach (string key in _httpContext.HttpContext!.Request.Form.Keys)
  106. sorted.Add(key, _httpContext.HttpContext.Request.Form[key]);
  107. string alipayPublicKey = Path.Combine(_webHostEnvironment.ContentRootPath, _alipayOptions.AlipayPublicCertPath!.Replace('/','\\').TrimStart('\\'));
  108. bool signVerified = AlipaySignature.RSACertCheckV1(sorted, alipayPublicKey, "UTF-8", _alipayOptions.SignType); // 调用SDK验证签名
  109. if (!signVerified) throw Oops.Oh("交易失败");
  110. var outTradeNo = sorted.GetValueOrDefault("out_trade_no");
  111. try
  112. {
  113. // 记录回调日志
  114. File.AppendAllText($"{_webHostEnvironment.ContentRootPath}\\AlipayLog\\Notify-{DateTime.Today:yyyy-MM-dd}.txt",
  115. $"支付宝支付到平台({DateTime.Now:yyyy-MM-dd HH:mm:ss}):{Environment.NewLine} " +
  116. $"登录人:{_userManager.UserId}-{_userManager.RealName}{Environment.NewLine} " +
  117. $"IP:{App.HttpContext?.GetRemoteIpAddressToIPv4(true)} {Environment.NewLine} " +
  118. $"交易号:{outTradeNo}{Environment.NewLine} " +
  119. $"参数:{JSON.Serialize(sorted)}{Environment.NewLine} " +
  120. $"-----------------------------------------------------------------------------------------------------------------------" +
  121. $"{Environment.NewLine}{Environment.NewLine}{Environment.NewLine}");
  122. }
  123. catch (Exception ex)
  124. {
  125. Log.Error("支付宝支付回调日志写入失败:", ex);
  126. }
  127. if (sorted.GetValueOrDefault(AlipayConst.TradeStatus) == AlipayConst.TradeSuccess)
  128. {
  129. // 约定交易码前四位为类型码,后面为订单号
  130. var tradeNo = long.Parse(outTradeNo);
  131. var type = long.Parse(outTradeNo.Substring(0, 4));
  132. // 循环执行业务逻辑,若都未处理(回调全部返回false)则交易失败
  133. var isError = true;
  134. foreach (var notify in App.GetServices<IAlipayNotify>())
  135. if (notify.TopUpCallback(type, tradeNo)) isError = false;
  136. if (isError) throw Oops.Oh("交易失败");
  137. }
  138. return "success";
  139. }
  140. /// <summary>
  141. /// 统一收单下单并支付页面接口
  142. /// </summary>
  143. /// <param name="input"></param>
  144. /// <returns></returns>
  145. [DisplayName("统一收单下单并支付页面接口")]
  146. [ApiDescriptionSettings(Name = "AlipayTradePagePay"), HttpPost]
  147. public string AlipayTradePagePay(AlipayTradePagePayInput input)
  148. {
  149. AlipayTradeWapPayRequest request = new();
  150. // 组装业务参数model
  151. AlipayTradeWapPayModel model = new()
  152. {
  153. Subject = input.Subject,
  154. OutTradeNo = input.OutTradeNo,
  155. TotalAmount = input.TotalAmount,
  156. Body = input.Body,
  157. ProductCode = "QUICK_WAP_WAY",
  158. TimeExpire = input.TimeoutExpress
  159. };
  160. request.SetBizModel(model);
  161. // 设置异步通知接收地址
  162. request.SetNotifyUrl(_alipayOptions.NotifyUrl);
  163. var response = _alipayClient.SdkExecute(request);
  164. if (response.IsError) throw Oops.Oh(response.SubMsg);
  165. return $"{_alipayOptions.ServerUrl}?{response.Body}";
  166. }
  167. /// <summary>
  168. /// 交易预创建
  169. /// </summary>
  170. /// <param name="input"></param>
  171. /// <returns></returns>
  172. [DisplayName("交易预创建")]
  173. [ApiDescriptionSettings(Name = "AlipayPreCreate"), HttpPost]
  174. public string AlipayPreCreate(AlipayPreCreateInput input)
  175. {
  176. AlipayTradePrecreateRequest request = new();
  177. // 设置异步通知接收地址
  178. request.SetNotifyUrl(_alipayOptions.NotifyUrl);
  179. // 组装业务参数model
  180. AlipayTradePrecreateModel model = new()
  181. {
  182. Subject = input.Subject,
  183. OutTradeNo = input.OutTradeNo,
  184. TotalAmount = input.TotalAmount,
  185. TimeoutExpress = input.TimeoutExpress
  186. };
  187. request.SetBizModel(model);
  188. var response = _alipayClient.CertificateExecute(request);
  189. if (response.IsError) throw Oops.Oh(response.SubMsg);
  190. return response.QrCode;
  191. }
  192. /// <summary>
  193. /// 单笔转账到支付宝账户
  194. /// https://opendocs.alipay.com/open/62987723_alipay.fund.trans.uni.transfer
  195. /// </summary>
  196. public Task<AlipayFundTransUniTransferResponse> Transfer(AlipayFundTransUniTransferInput input)
  197. {
  198. // 构造请求参数以调用接口
  199. AlipayFundTransUniTransferRequest request = new();
  200. AlipayFundTransUniTransferModel model = new();
  201. model.BizScene = AlipayConst.BizScene;
  202. model.ProductCode = AlipayConst.ProductCode;
  203. // 设置商家侧唯一订单号
  204. model.OutBizNo = input.OutBizNo;
  205. // 设置订单总金额
  206. model.TransAmount = input.TransAmount.ToString();
  207. // 设置转账业务的标题
  208. model.OrderTitle = input.OrderTitle;
  209. // 设置收款方信息
  210. Participant payeeInfo = new();
  211. payeeInfo.CertType = input.CertType.ToString();
  212. payeeInfo.CertNo = input.CertNo;
  213. payeeInfo.Identity = input.Identity;
  214. payeeInfo.Name = input.Name;
  215. payeeInfo.IdentityType = input.IdentityType.ToString();
  216. model.PayeeInfo = payeeInfo;
  217. // 设置业务备注
  218. model.Remark = input.Remark;
  219. // 设置转账业务请求的扩展参数
  220. string payerShowNameUseAlias = input.PayerShowNameUseAlias.ToString().ToLower();
  221. model.BusinessParams = $"{{\"payer_show_name_use_alias\":\"{payerShowNameUseAlias}\"}}";
  222. request.SetBizModel(model);
  223. var response = _alipayClient.CertificateExecute(request);
  224. try
  225. {
  226. File.AppendAllText($"{_webHostEnvironment.ContentRootPath}\\AlipayLog\\{DateTime.Today:yyyy-MM-dd}.txt",
  227. $"支付宝付款到账户({DateTime.Now:yyyy-MM-dd HH:mm:ss}):{Environment.NewLine} " +
  228. $"登录人:{_userManager.UserId}-{_userManager.RealName}{Environment.NewLine} " +
  229. $"IP:{App.HttpContext?.GetRemoteIpAddressToIPv4(true)} {Environment.NewLine} " +
  230. $"参数:{JSON.Serialize(model)}{Environment.NewLine} " +
  231. $"返回:{JSON.Serialize(response)}{Environment.NewLine}" +
  232. $"-----------------------------------------------------------------------------------------------------------------------" +
  233. $"{Environment.NewLine}{Environment.NewLine}{Environment.NewLine}");
  234. }
  235. catch(Exception ex)
  236. {
  237. Log.Error("单笔转账到支付宝账户日志写入失败:", ex);
  238. }
  239. return Task.FromResult(response);
  240. }
  241. }