AlipayService.cs 11 KB

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