AlipayService.cs 11 KB

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