飞道的博客

PHP对接支付宝当面付详细教程

342人阅读  评论(0)

导读:

作为一名小站长或开发者,网站需要接入支付功能,才能实现网站的变现盈利,一般有以下几个方案:

1、第三方支付平台: 截止2020年11月23日,我国拥有第三方支付牌照的公司共有237家,从2015年至今,先后有34家第三方支付公司支付牌照被注销。第三方支付不容易申请,基本都需要企业资格才能开通,审核比较严。

2020年第三方支付公司排名TOP50

排名 第三方支付公司 APP或支付品牌
1 财付通支付科技有限公司 财付通和微信支付
2 支付宝(中国)网络技术有限公司 支付宝
3 中国银联股份有限公司 银联商务
4 平安付科技服务有限公司 壹钱包
5 快钱支付清算信息有限公司 快钱
6 苏宁消费金融有限公司 苏宁金融(苏宁支付)
7 联动优势电子商务有限公司 联动优势
8 京东数字科技控股股份有限公司 京东支付
9 拉卡拉支付股份有限公司 拉卡拉
10 通联支付网络服务股份有限公司 通联支付
11 易宝支付有限公司 易宝支付(Yeepay)
12 迅付信息科技有限公司 环迅支付
13 北京度小满支付科技有限公司 度小满支付
14 天翼电子商务有限公司 翼支付
15 中移电子商务有限公司 和包支付
16 网银在线(北京)科技有限公司 网银在线
17 联通支付有限公司 联通支付
18 上海盛付通电子支付服务有限公司 盛付通
19 北京海科融通支付服务股份有限公司 海科融通
20 易生支付有限公司 易生支付
21 随行付支付有限公司 随行付Plus
22 海南海岛一卡通支付网络有限公司 海南一卡通
23 国付宝信息科技有限公司 国付宝
24 银盛支付服务股份有限公司 银盛支付
25 网易宝有限公司 网易宝
26 连连银通电子支付有限公司 连连支付
27 北京钱袋宝支付技术有限公司 钱袋宝
28 宝付网络科技(上海)有限公司 宝付
29 杉德支付网络服务发展有限公司 杉德支付
30 智付电子支付有限公司 智付Dinpay
31 中付支付科技有限公司 中付支付
32 上海付费通信息服务有限公司 付费通
33 汇付天下有限公司 汇付天下
34 瑞银信支付技术有限公司 瑞银信
35 北京新浪支付科技有限公司 新浪支付
36 上海富友支付服务股份有限公司 富友支付
37 易智付科技(北京)有限公司 首信易支付
38 快捷通支付服务有限公司 快捷通
39 中金支付有限公司 中金支付
40 广州市汇聚支付电子科技有限公司 汇聚支付
41 得仕股份有限公司 得仕通
42 资和信电子支付有限公司 资和信
43 易联支付有限公司 易联支付
44 上海偶可贝网络科技有限公司 Allpay
45 优钱付(浙江)信息科技有限公司 优钱付
46 四川商通实业有限公司 四川商通
47 卡友支付服务有限公司 卡友支付
48 重庆易极付科技有限公司 易极付
49 双乾网络支付有限公司 双乾支付
50 新生支付有限公司 新生支付

2、第四方支付(聚合支付):第四方支付是相对第三方而言的,作为对第三方支付平台服务的拓展,第三方支付介于银行和商户之间,而第四方支付是介于第三方支付和商户之间,没有支付许可牌照的限制。第三方支付提供的是资金清算通道,而第四方支付提供的是支付基础上的多种衍生服务。作为连接着第三方支付机构和商户的中间商,聚合支付只是完成支付环节的信息流转和商户操作的承载,从事的是“支付、结算、清算”服务之外的“支付服务”,并不提供资金清算通道。《中国人民银行支付结算司关于开展违规“聚合支付”服务清理整治工作的通知》,将聚合支付定位于“收单外包机构”,对聚合支付划了四条明确的红线,即“四个不得”,其中之一是“不得以任何形式经手特约商户结算资金,从事或变相从事特约商户资金结算”。换句话理解,聚合支付公司只是为商户提供第三方支付通道的技术服务机构,不能沉淀资金,更不能为商户提供支付和资金清算。聚合支付本身并不违法,但如果聚合支付平台从事了结算业务,对商家的资金进行了截留,形成所谓“资金池”,就是非法行为,风险很大。正是因为风险的存在,所以很多未经国家有关主管部门批准的小型聚合支付平台都是做一段时间就跑路,就是为了规避这个风险。大家要选择第四方聚合支付平台时,一定要谨慎!

3、个人免签约系统:使用个人的支付宝或微信收款码,然后安卓监听APP收款通知,实现收款。本质上是采用挂机监听的策略,但针对的是移动端支付宝或微信的收款通知消息,成本高,配置麻烦,需24小时挂台安卓手机,不免费。

对于个人开发者来说,以上方案要么没有企业资质,要么太麻烦,要么太贵,有没有更好的方案呢?答案是:当然有,那就是支付宝的当面付,下面让我们一步一步来探讨。

当面付申请条件:

1.  支持的账户类型:经过实名认证的个人/企业支付宝账号  

2.  签约申请提交资料: 

   1) 经营场所照片

     a. 有店铺门头的经营场所,需提供门头照;

     b. 无店铺门头的经营场所,需提供内景照或场景照

提醒:若未规范提交经营场景照片,商家收款将受限,商家需在30天内补全资料,否则将影响正常收款。

   2) 与实名认证的支付宝账号持有人同名的营业执照(即营业执照的法人代表与支付宝账号持有人姓名一致)

提醒:若未提供同名营业执照,可长期使用但商家收款有限额。

3.费率

服务名称 费率 服务期限
单笔费率 0.6% 1年

说明:

  • 申请当面付其实很简单,只需要找一张店铺门面照片即可,可以到外面随便拍一张,或者网站搜索。
  • 营业执照是可选的,不上传的话,限制单笔收款≤1000,单日收款≤5W,对于小网站来说,这个额度已经够用了。

补充:

如果你的支付宝账号在申请时,提示风险过高,那就换另一个账号。支付宝的规则系统一般会检测你账号近期的交易,看是否有风险,新注册的支付宝账号,一般都申请不了当面付。

当面付申请流程:

1、点击这里进入,登陆支付宝账户,然后点击“立即开通”。

2、填写基本信息,红色星号的必填:

  • 经营内容选择百货零售 → 超市 → 超市,或者选其他的,建议不要选平台类、支付类、游戏类的,这些审查严格,没有企业资质和电信增值许可证是通过不了。
  • 营业执照可以不上传。
  • 店铺招牌,就是门店的照片,可以去外面拍一张,也可以百度搜索。
  • 提交申请后十多分钟就可收到通过通知。

当面付开发流程:

一、准备工作

1、首先是要申请成功开发者,可以登录https://developers.alipay.com/申请。

2、当面付申请成功后,可以在蚂蚁金服开放平台网页&移动应用中,看到我的应用列表中多了一个 “应用2.0签约20210203*********” 的应用

二、配置当面付公钥和私钥

1、利用 “支付宝开放平台开发助手” 生成RSA密钥(包括应用公钥和应用私钥),官方教程地址:https://opendocs.alipay.com/open/291/105971

2、点击进入应用2.0签约20210203*********

3、在 左侧菜单栏的“应用信息” 中设置公钥,设置“接口加签方式”

4、把前面生成的应用公钥复制进去,然后保存设置即可

三、官方API参数

网关地址:https://openapi.alipay.com/gateway.do

1、公共请求参数:

参数 类型 是否必填 最大长度 描述
app_id String 32 支付宝分配给开发者的应用ID
method String 128 接口名称
format String 40 仅支持JSON
charset String 10 请求使用的编码格式,如utf-8,gbk,gb2312等
sign_type String 10 商户生成签名字符串所使用的签名算法类型,目前支持RSA
sign String 256 商户请求参数的签名串,详见签名
timestamp String 19 发送请求的时间,格式"yyyy-MM-dd HH:mm:ss"
version String 3 调用的接口版本,固定为:1.0
notify_url String 256 支付宝服务器主动通知商户服务器里指定的页面http/https路径。
app_auth_token String 40 详见应用授权概述
biz_content String - 请求参数的集合,最大长度不限,除公共参数外所有请求参数都必须放在这个参数中传递,具体参照各产品快速接入文档

2、请求参数:

参数 类型 是否必填 最大长度 描述
out_trade_no String 必须 64 商户订单号,64个字符以内、可包含字母、数字、下划线;需保证在商户端不重复
scene String 必须 32 支付场景 条码支付,取值:bar_code 声波支付,取值:wave_code
auth_code String 必须 32 支付授权码
seller_id String 可选 28 如果该值为空,则默认为商户签约账号对应的支付宝用户ID
total_amount Price 可选 11 订单总金额,单位为元,
discountable_amount Price 可选 11 参与优惠计算的金额,单位为元
undiscountable_amount Price 可选 11 不参与优惠计算的金额,单位为元
subject String 必须 256 订单标题
body String 可选 128 订单描述
goods_detail GoodsDetail [] 可选 - 订单包含的商品列表信息,Json格式,其它说明详见商品明细说明
operator_id String 可选 28 商户操作员编号
store_id String 可选 32 商户门店编号
terminal_id String 可选 32 商户机具终端编号
alipay_store_id String 可选 32 支付宝的店铺编号
extend_params ExtendParams 可选 - 业务扩展参数
timeout_express String 可选 6 该笔订单允许的最晚付款时间,逾期将关闭交易。
royalty_info RoyaltyInfo 可选 - 描述分账信息,Json格式,其它说明详见分账说明
sub_merchant SubMerchant 可选 - 二级商户信息,当前只对特殊银行机构特定场景下使用此字段

3、公共响应参数:

参数 类型 是否必填 最大长度 描述
code String - 网关返回码,详见文档
msg String - 网关返回码描述,详见文档
sub_code String - 业务返回码,详见文档
sub_msg String - 业务返回码描述,详见文档
sign String - 签名,详见文档

4、响应参数

参数 类型 是否必填 最大长度 描述
trade_no String 必填 64 支付宝交易号
out_trade_no String 必填 64 商户订单号
buyer_logon_id String 必填 100 买家支付宝账号
total_amount Price 必填 11 交易金额
receipt_amount String 必填 11 实收金额
buyer_pay_amount Price 选填 11 买家付款的金额
point_amount Price 选填 11 使用积分宝付款的金额
invoice_amount Price 选填 11 交易中可给用户开具发票的金额
gmt_payment Date 必填 32 交易支付时间
fund_bill_list TradeFundBill [] 必填 - 交易支付使用的资金渠道
card_balance Price 选填 11 支付宝卡余额
store_name String 选填 512 发生支付交易的商户门店名称
buyer_user_id String 必填 28 买家在支付宝的用户id
discount_goods_detail String 必填 - 本次交易支付所使用的单品券优惠的商品优惠信息

三、沙箱测试环境

如果当面付暂时还没有签约成功,可以先使用沙箱账号进行测试,方法如下:

1、登录支付宝开发平台 https://openhome.alipay.com/platform/home.htm,开发服务 → 研发服务

2、沙箱应用里,设置RSA2(SHA256)密钥,和上面设置应用公钥是一样的。

3、设置应用网关,用于接收支付宝异步通知

四、 代码编写

可以下载官方当面付的SDK示例,下载链接:https://opendocs.alipay.com/open/194/105201/

我大概地看了一下官方给的示例,涉及的类文件比较多,对于新手而言还是比较复杂的,所以我简化了一下,大家可以参考我以下的Thinkphp框架代码。

1、在extend扩展目录下新建一个pay文件夹,里面创建一个Alipay.php类,代码如下:


  
  1. <?php
  2. namespace pay;
  3. use \ think\ Db;
  4. use app\ common\ model\ Order as OrderModel;
  5. /**
  6. * 支付宝支付类
  7. */
  8. class Alipay {
  9. //是否沙盒环境
  10. public $is_sandbox = true;
  11. //沙盒地址
  12. private $sandurl = 'https://openapi.alipaydev.com/gateway.do';
  13. //正式地址
  14. private $apiurl = 'https://openapi.alipay.com/gateway.do';
  15. //网关地址
  16. private $gateway;
  17. //支付宝的APPID
  18. private $appid;
  19. //应用私钥
  20. private $rsaPrivateKey = '商户设置的私钥';
  21. //支付宝公钥
  22. private $alipayPublicKey= '支付宝自动生成的公钥';
  23. private $charset = 'utf-8';
  24. public function setAppid($appid)
  25. {
  26. $this->appid = $appid;
  27. }
  28. public function setRsaPrivateKey($rsaPrivateKey)
  29. {
  30. $this->rsaPrivateKey = $rsaPrivateKey;
  31. }
  32. public function setAlipayPublicKey($alipayPublicKey)
  33. {
  34. $this->alipayPublicKey = $alipayPublicKey;
  35. }
  36. //构造方法
  37. public function __construct($account){
  38. //如果是沙箱测试
  39. if( $this->is_sandbox){
  40. $this->gateway = $this->sandurl;
  41. $this->appid = '2021000117612368';
  42. $this->rsaPrivateKey = 'MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQDF0Bwt0712ogHbqp245rlz6Rj7LePxxhO4HoIUZhmZS7OeC1muhLK44e9kznFGT6YvOE2rasZT+EixPLifYqUML0Ecmxxvt3dr/WrAIIM941ODhjCaV+CDjZYNBZmsK+nPojFNWuyZCq9ygzX43XOjk3oTNbws5/tJdA4fzKIhlufuvpeGL5tgzoLKxUy2oweVluIyqkhFZUs9/9vgiUwFVt/iPf4JREGJ56Vq1grOBd+ZGqrljCcx6mEOm7GGd0lcDw038L//MYICdAgLfmCF+aojkRm3qkcTt7bl0jYsYnAWqxkBvtpCWQzFKMbwVsqbOM5EUzle8csNuguXqZ3TAgMBAAECggEAQXoSacTluOgFT23ZSUPt7QUiRLo68mOLUW2fxxrIrGpXaR/+rnq0Ieh/clG3QvvIWWb8ppnwTgkoHqqnvfJMCY12sv5Q2lXTTMaUWEYqywh9M1LusZgW89Sg72yEIm5itVzNjvpiyIG182SlL9w1k9aUGNm3kPG0xuqIZlQ48n7IZSeDGg+5ic4J+U7PGaCV99Q2zq7mkuIrbot+p478oFR7aM1/24OfxdJ1eh4FZZevwmLHpwqZEhecZFGrKx6/wlzDOhaiR7PgGcjH0/q6tiw6g5TCWfAzBDfFl/ZdRToNP6mOe57qt7jKhQz+2Lxz0qLNdNDacyTSYIkFg9cE0QKBgQD3AyFhrFk0VaTN2Xt6OAOEn1UIf1aBPenFQfV8+tIlbL5Io3VHHmycMuctW7c2BGkp7HF9OwcYGtURQdaBRCyKAYnW1nbZRgLHhkFC9rLyG04/yv6pN6/8EZ4bow7aUiCGoR/mV5GehiWTjb5/thCS+enaaalTdAfQBGZOl7VyGwKBgQDNArKycIbtXl2BEqcIqwW2nzv0RA8FhK7LkRy1CFIAeZfr1O4FiSXM/3V/EuocISCwOLWRq5Qyx7mZoMZ4FSRlQFbo9LUmYtIXtKHekwYZEpxleXJOVudRRQa3GkMvWAd3s1MSCGQ44jHFc/vbap6fxfR/GWHupoMOvW4N06h+qQKBgAFzekG5oniFPEedTHVmWNbxnK1FGjv+Ih5vicKnMo7XubJxi2HUkuzD+8mvbT8a/YcqhWwn6Z3BopjMWzc9MEnLQoUJk38pQyDq7/01t81mTapgRei2lAkWDWi6J38u1lQUJDzVLNzSiuv3/DOB3U0PvMj0r/L2jokxTWyOxZkbAoGBAK8DuqHyxn0lrhVeYxJXTg1Vas1gvKItXzuRqjwx2i9lnnhJ0tkO+CJFg3z0HL/e2BUYlIjDPUUMlDm+szAYfHWjs440OeGHQ2vRXM6yHOaSqMlIIHkYX2lV0CHIXcqxD870W9ptJ4IYN/0kwsHSB5DGYa+Pb7dYl1GiDa3oH1PhAoGBAPCSjFfXd45BPYDNL+4N9zS2wT5pjxlSo3Jrg3NKOqBZLS52ygZ6xeANXrG7lzitJ75prUS/JyXS83SM8au0vcO1Fko9BbW7MkuKFZH7FSpOlv5/PJQ6tI55TV7gGl8l4ND7V9OSdAMhEBQ/uT2gmddFoRTB14MeLIpe2iIVonan';
  43. $this->alipayPublicKey = 'MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAs9AE75V+F242HcN/SlcGUxUNv3zGpXjai4u2ZMkOL1nolr4/v4BGEsE8tLaYq6lecG/of5taQ27WbWXB5PMB3s6emesTEn5yvZPh/HqmKACHi7js+Dtu123J/z0DyCukMm1ZHJe7jT9VEU9w4gIdNi64VxdaRF3ZMT6DXnboUEegyRKGmJc58h+O30P/UYqdCW/Gl+380o80e6Fs0rX33AixkNTRNgnQ3n2er1Nrqan/9sfUCnqqxgpc1+GTT+vyn3x4Xwvch2pRIkcK4BFCrWVVMKVXx/icj1njTVwWm1KYrXDVYrxf4ZLgpMHyW3SfhnXFKuxvSYnvm5EHjxtNEQIDAQAB';
  44. } else{
  45. $this->gateway = $this->apiurl;
  46. $this->appid = $account[ 'appid'];
  47. $this->rsaPrivateKey = $account[ 'privateKey'];
  48. $this->alipayPublicKey = $account[ 'publicKey'];
  49. }
  50. }
  51. /**
  52. *
  53. * 发起当面付
  54. * $params 传输的数据
  55. */
  56. public function facepay($params){
  57. //请求参数的集合,参考https://mp.csdn.net/editor/html/113599448
  58. $biz_content = [
  59. "scene" => "bar_code", //支付场景 条码支付,取值:bar_code 声波支付,取值:wave_code(必须)
  60. "out_trade_no" =>$params[ 'pay_id'], //唯一标识,订单编号(必须)
  61. "total_amount" =>$params[ 'money'], //订单金额(可选)
  62. "subject" => "购买商品", //商品名称
  63. //"seller_id" =>'', //如果该值为空,则默认为商户签约账号对应的支付宝用户ID
  64. "timeout_express" => "5m", //该笔订单允许的最晚付款时间,逾期将关闭交易。取值范围:1m~15d。m-分钟,h-小时,d-天
  65. ];
  66. //公共请求参数
  67. $param = [
  68. 'app_id' => $this->appid, //支付宝分配给开发者的应用ID
  69. 'method' => 'alipay.trade.precreate', //接口名称
  70. //'format' => 'JSON',//仅支持JSON
  71. 'charset' => 'utf-8', //请求使用的编码格式
  72. 'sign_type' => 'RSA2', //商户生成签名字符串所使用的签名算法类型,目前支持RSA2和RSA,推荐使用RSA2
  73. 'sign' => '', //商户请求参数的签名串
  74. 'timestamp' => date( 'Y-m-d H:i:s', time()), //发送请求的时间,格式"yyyy-MM-dd HH:mm:ss"
  75. 'version' => '1.0', //调用的接口版本,固定为1.0
  76. 'notify_url' => $params[ 'notify_url'], //异步通知地址,支付宝服务器主动通知商户服务器里指定的页面http/https路径
  77. //'app_auth_token' => '',//app_auth_token
  78. 'biz_content' => json_encode($biz_content), //请求参数的集合,最大长度不限,除公共参数外所有请求参数都必须放在这个参数中传递
  79. ];
  80. //组合生成签名参数
  81. $signdata = [];
  82. $signdata[ 'app_id'] = $param[ 'app_id'];
  83. $signdata[ 'method'] = $param[ 'method'];
  84. $signdata[ 'charset'] = $param[ 'charset'];
  85. $signdata[ 'sign_type'] = $param[ 'sign_type'];
  86. $signdata[ 'timestamp'] = $param[ 'timestamp'];
  87. $signdata[ 'version'] = $param[ 'version'];
  88. $signdata[ 'notify_url'] = $param[ 'notify_url'];
  89. $signdata[ 'biz_content'] = $param[ 'biz_content'];
  90. //生成签名
  91. $sign = $this->generateSign($signdata, 'RSA2');
  92. $param[ 'sign'] = $sign;
  93. //echo "<pre>";
  94. //var_dump($param);die;
  95. //发起请求
  96. $content = $this->curlPost( $this->gateway,$param);
  97. $alipayData = json_decode($content, true);
  98. //公共响应参数
  99. $responseData = $alipayData[ 'alipay_trade_precreate_response'];
  100. if($responseData[ 'code'] == 10000){
  101. //生成成功,返回结果给前端
  102. $data = [];
  103. $data[ 'out_trade_no'] = $responseData[ 'out_trade_no'];
  104. $data[ 'qr_code'] = $responseData[ 'qr_code'];
  105. return [ 'code' => 1 , 'msg' => '成功' , 'data' => $data];
  106. } else {
  107. //file_put_contents(LOG_PATH .'alipayFacepay.log', 'err code:' . $responseData['code'] . ', err msg:' . $responseData['msg'] . '\r\n', FILE_APPEND);
  108. return [ 'code' => 0 , 'msg' => '错误码:' . $responseData[ 'code'] . ',错误信息:' . $responseData[ 'msg']];
  109. }
  110. }
  111. /**
  112. * 支付宝支付通知
  113. * @param $data 通知的数据
  114. */
  115. public function notify($data){
  116. $falg= false;
  117. //验证签名
  118. $param = $data;
  119. unset($param[ 'sign']);
  120. unset($param[ 'sign_type']);
  121. $rst = $this->rsaCheck($param, $data[ 'sign'] , $data[ 'sign_type']);
  122. if(!$rst){
  123. file_put_contents(LOG_PATH . 'alipaynotify.log', '验签失败\r\n' , FILE_APPEND );
  124. return false;
  125. }
  126. //查询支付订单状态
  127. try{
  128. $rst = $this->orderquery($data, 'TRADE_SUCCESS');
  129. } catch (\ Exception $e) {
  130. printLog( "查询支付订单状态失败:".$e);
  131. }
  132. if($rst){
  133. $falg= true;
  134. } else {
  135. file_put_contents(LOG_PATH . 'alipaynotify.log', '查询订单状态错误\r\n', FILE_APPEND);
  136. $falg= false;
  137. }
  138. return $falg;
  139. }
  140. /**
  141. *
  142. * 支付查询接口
  143. * @param data 支付宝响应的参数集合
  144. * @param status 要验证的状态
  145. * WAIT_BUYER_PAY 交易创建等待买家付款
  146. * TRADE_CLOSED 未付款交易超时关闭或支付完成后全额退款
  147. * TRADE_SUCCESS 交易支付成功
  148. * TRADE_FINISHED 交易结束不可退款
  149. */
  150. public function orderquery($data , $status){
  151. $biz_content = [
  152. 'out_trade_no' => $data[ 'out_trade_no'],
  153. 'trade_no' => $data[ 'trade_no'],
  154. //'org_pid' => '',
  155. ];
  156. $param = [
  157. 'app_id' => $this->appid,
  158. 'method' => 'alipay.trade.query',
  159. 'charset' => 'utf-8',
  160. 'sign_type' => 'RSA2',
  161. 'sign' => '',
  162. 'timestamp' => date( 'Y-m-d H:i:s', time()),
  163. 'version' => '1.0',
  164. 'biz_content' => json_encode($biz_content),
  165. ];
  166. //组合签名数组
  167. $signdata = [];
  168. $signdata[ 'app_id'] = $param[ 'app_id'];
  169. $signdata[ 'method'] = $param[ 'method'];
  170. $signdata[ 'charset'] = $param[ 'charset'];
  171. $signdata[ 'sign_type'] = $param[ 'sign_type'];
  172. $signdata[ 'timestamp'] = $param[ 'timestamp'];
  173. $signdata[ 'version'] = $param[ 'version'];
  174. $signdata[ 'biz_content'] = $param[ 'biz_content'];
  175. //生成签名
  176. $sign = $this->generateSign($signdata, 'RSA2');
  177. $param[ 'sign'] = $sign;
  178. $content = $this->curlPost( $this->gateway,$param);
  179. $alipayData = json_decode($content, true);
  180. //公共响应参数
  181. $responseData = $alipayData[ 'alipay_trade_query_response'];
  182. if($responseData[ 'code'] == 10000){
  183. if($responseData[ 'trade_status'] == $status){
  184. return true;
  185. } else {
  186. return false;
  187. }
  188. } else {
  189. return false;
  190. }
  191. }
  192. public function generateSign($params, $signType = "RSA") {
  193. return $this->sign( $this->getSignContent($params), $signType);
  194. }
  195. public function getSignContent($params) {
  196. ksort($params);
  197. $stringToBeSigned = "";
  198. $i = 0;
  199. foreach ($params as $k => $v) {
  200. if ( false === $this->checkEmpty($v) && "@" != substr($v, 0, 1)) {
  201. // 转换成目标字符集
  202. $v = $this->characet($v, $this->charset);
  203. if ($i == 0) {
  204. $stringToBeSigned .= "$k" . "=" . "$v";
  205. } else {
  206. $stringToBeSigned .= "&" . "$k" . "=" . "$v";
  207. }
  208. $i++;
  209. }
  210. }
  211. unset ($k, $v);
  212. return $stringToBeSigned;
  213. }
  214. /**
  215. * 转换字符集编码
  216. * @param $data
  217. * @param $targetCharset
  218. * @return string
  219. */
  220. function characet($data, $targetCharset) {
  221. if (! empty($data)) {
  222. $fileType = $this->charset;
  223. if (strcasecmp($fileType, $targetCharset) != 0) {
  224. $data = mb_convert_encoding($data, $targetCharset, $fileType);
  225. //$data = iconv($fileType, $targetCharset.'//IGNORE', $data);
  226. }
  227. }
  228. return $data;
  229. }
  230. /**
  231. *
  232. * 校验$value是否非空
  233. */
  234. protected function checkEmpty($value) {
  235. if (! isset($value))
  236. return true;
  237. if ($value === null)
  238. return true;
  239. if (trim($value) === "")
  240. return true;
  241. return false;
  242. }
  243. /**
  244. *
  245. * 签名函数
  246. */
  247. protected function sign($data, $signType = "RSA") {
  248. $priKey= $this->rsaPrivateKey;
  249. $res = "-----BEGIN RSA PRIVATE KEY-----\n" .
  250. wordwrap($priKey, 64, "\n", true) .
  251. "\n-----END RSA PRIVATE KEY-----";
  252. ($res) or die( '您使用的私钥格式错误,请检查RSA私钥配置');
  253. if ( "RSA2" == $signType) {
  254. //OPENSSL_ALGO_SHA256是php5.4.8以上版本才支持
  255. openssl_sign($data, $sign, $res, version_compare(PHP_VERSION, '5.4.0', '<') ? SHA256 : OPENSSL_ALGO_SHA256);
  256. } else {
  257. openssl_sign($data, $sign, $res);
  258. }
  259. $sign = base64_encode($sign);
  260. return $sign;
  261. }
  262. /**
  263. *
  264. * 验签函数
  265. */
  266. public function rsaCheck($data, $sign,$type = 'RSA'){
  267. $public_key = $this->alipayPublicKey;
  268. $search = [
  269. "-----BEGIN PUBLIC KEY-----",
  270. "-----END PUBLIC KEY-----",
  271. "\n",
  272. "\r",
  273. "\r\n"
  274. ];
  275. $public_key=str_replace($search, "",$public_key);
  276. $public_key=$search[ 0] . PHP_EOL . wordwrap($public_key, 64, "\n", true) . PHP_EOL . $search[ 1];
  277. $res=openssl_get_publickey($public_key);
  278. if($res)
  279. {
  280. if($type == 'RSA'){
  281. $result = ( bool)openssl_verify( $this->getSignContent($data), base64_decode($sign), $res);
  282. } elseif($type == 'RSA2'){
  283. $result = ( bool)openssl_verify( $this->getSignContent($data), base64_decode($sign), $res,OPENSSL_ALGO_SHA256);
  284. }
  285. openssl_free_key($res);
  286. } else{
  287. return false;
  288. }
  289. return true;
  290. }
  291. /*
  292. * curl发送post请求,并返回请求头信息
  293. * url: 访问路径
  294. * postData: 要传递的post数据
  295. * refcode: 是否返回请求码
  296. * refheader: 是否返回请求头信息
  297. * */
  298. function curlPost($url, $postData, $refcode = false, $refheader = false) {
  299. $curl = curl_init();
  300. //设置提交的url
  301. curl_setopt($curl, CURLOPT_URL, $url);
  302. //设置获取的信息以文件流的形式返回,而不是直接输出
  303. curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
  304. //忽略证书(关闭https验证)
  305. curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
  306. curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
  307. //设置post方式提交
  308. curl_setopt($curl, CURLOPT_POST, 1);
  309. //设置post数据
  310. $postFields = http_build_query($postData);
  311. curl_setopt($curl, CURLOPT_POSTFIELDS, $postFields);
  312. //添加请求头信息
  313. $headers = addHttpHeader($url);
  314. curl_setopt($curl, CURLOPT_HTTPHEADER, $headers);
  315. //在尝试连接时等待的秒数
  316. curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 30);
  317. //设置超时时间,最大执行时间超时时间(单位:s)
  318. curl_setopt($curl, CURLOPT_TIMEOUT, 300);
  319. //是否返回请求头信息(http协议头)
  320. if ($refheader) {
  321. curl_setopt($curl, CURLOPT_HEADER, 1);
  322. //追踪句柄的请求字符串(允许查看请求header)
  323. curl_setopt($curl, CURLINFO_HEADER_OUT, true);
  324. } else {
  325. curl_setopt($curl, CURLOPT_HEADER, 0);
  326. }
  327. //执行命令
  328. $result = curl_exec($curl);
  329. //转换字符编码
  330. $result = mb_convert_encoding($result, 'utf-8', 'UTF-8,GBK,GB2312,BIG5');
  331. //解决返回的json字符串中返回了BOM头的不可见字符(某些编辑器默认会加上BOM头)
  332. $result = trim($result,chr( 239).chr( 187).chr( 191));
  333. //获取状态码
  334. $httpcode = curl_getinfo($curl, CURLINFO_HTTP_CODE);
  335. //关闭URL请求
  336. curl_close($curl);
  337. //是否返回请求码
  338. if ($refcode) {
  339. $html = array( "httpcode" => $httpcode, "result" => $result);
  340. return $html;
  341. } else {
  342. return $result;
  343. }
  344. }
  345. }
  346. ?>

2、在app\common目录创建一个Pay.php支付类,代码如下:


  
  1. <?php
  2. namespace app\ common;
  3. use think\ Db;
  4. use think\ Exception;
  5. /**
  6. * 支付类
  7. */
  8. class Pay {
  9. public $account = array(
  10. "appid" => "支付宝的APPID",
  11. "privateKey" => "商户设置的私钥",
  12. "publicKey" => "支付宝自动生成的公钥"
  13. );
  14. public function startPay($params){
  15. //创建支付宝对象
  16. $alipayobj= new \pay\Alipay( $this->account);
  17. $result=$alipayobj->facepay($params);
  18. ($result[ 'code']== 0) && exit( '支付通道维护,'.$result[ 'msg']);
  19. //接收支付宝返回的订单编号和支付二维码
  20. $params[ 'out_trade_no'] = $result[ 'data'][ 'out_trade_no'];
  21. $params[ 'qr_code'] = $result[ 'data'][ 'qr_code'];
  22. return $this->buildRequestForm(url( '@Index/Index/facepay'),$params);
  23. }
  24. /**
  25. * 服务器回调
  26. */
  27. public function notify_callback($params){
  28. //查询数据库订单状态
  29. $order = OrderModel::where( 'number', $params[ 'out_trade_no'])->find();
  30. if(!$order){
  31. die( '不存在编号['.$params[ 'out_trade_no']. ']的订单!');
  32. }
  33. //防止恶意刷新加钱
  34. if ($order[ 'status'] == 1) {
  35. //直接返回给上游 success 或者 OK 之类的
  36. exit( 'success');
  37. }
  38. //创建支付宝对象
  39. $alipayobj= new \pay\Alipay( $this->account);
  40. //调用异步通知验证
  41. $verifyresult=$alipayobj->notify($params);
  42. if ($verifyresult) {
  43. //实际付款金额
  44. $money = ( float)$params[ 'total_amount'];
  45. //这里完成网站的业务逻辑
  46. exit( 'success');
  47. } else{
  48. exit( 'fail'); //返回失败 继续补单
  49. }
  50. }
  51. /**
  52. * 建立跳转请求表单
  53. * @param string $url 数据提交跳转到的URL
  54. * @param array $data 请求参数数组
  55. * @param string $method 提交方式:post或get 默认post
  56. * @return string 提交表单的HTML文本
  57. */
  58. function buildRequestForm($url, $data, $method = 'post', $button_name = '确定', $show = false) {
  59. $html = "<form id='requestForm' name='requestForm' action='" . $url . "' method='" . $method . "'>";
  60. while ( list($key, $val) = each($data)) {
  61. $html.= "<input type='hidden' name='" . $key . "' value='" . $val . "' />";
  62. }
  63. $display = $show ? "style='display:block;'" : "style='display:none;'";
  64. $html.= "<input type='submit' value='" . $button_name . "' " . $display . "></form>";
  65. $html.= "<script>document.forms['requestForm'].submit();</script>";
  66. return $html;
  67. }
  68. }
  69. ?>

3、controller控制器里发起支付请求


  
  1. <?php
  2. namespace app\ index\ controller;
  3. use think\ Controller;
  4. use think\ Cache;
  5. use think\ Session;
  6. use think\ Request;
  7. use app\ common\ Pay;
  8. class Index extends Controller
  9. {
  10. //接口提交
  11. public function submit() {
  12. $array= array(
  13. "pay_id" => "订单编号",
  14. "money" => "订单金额",
  15. "notify_url" => "异步通知地址",
  16. );
  17. $res = ( new Pay)->startPay($array);
  18. exit($res);
  19. }
  20. //当面付页面
  21. public function facepay(){
  22. $data=input( '');
  23. ! isset($data[ 'pay_id']) && $this->error( '订单编号不能为空');
  24. empty(trim($data[ 'pay_id'])) && $this->error( '订单编号不能为空');
  25. ! isset($data[ 'money']) && $this->error( '金额不能为空');
  26. empty(trim($data[ 'money'])) && $this->error( '金额不能为空');
  27. ! isset($data[ 'notify_url']) && $this->error( '异步通知地址不能为空');
  28. empty(trim($data[ 'notify_url'])) && $this->error( '异步通知地址不能为空');
  29. //生成二维码
  30. $qrcode= 'https://my.tv.sohu.com/user/a/wvideo/getQRCode.do?width=200&height=200&text='.trim($data[ 'qr_code']);
  31. $this->assign( 'money',trim($data[ 'money']));
  32. $this->assign( 'qrcode',$qrcode);
  33. return view();
  34. }
  35. }

到这里其实已经差不多了,但还有几个功能需要完成:

  • 显示当面付二维码的页面facepay.html,即facepay()方法的视图。
  • 支付宝当面付是没有同步通知功能的,所以要自己解决,解决思路就是在facepay.html页面使用js定时器,定时查询订单状态,如果订单完成,就跳转到同步通知页面。

最后附上我开发的效果图:

 


转载:https://blog.csdn.net/qq15577969/article/details/113599448
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场