SdkPayment.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. <?php
  2. namespace app\common\components\alipay\Mobile;
  3. use app\common\events\PayLog;
  4. use app\common\services\alipay\MobileAlipay;
  5. class SdkPayment
  6. {
  7. private $__https_verify_url = 'https://mapi.alipay.com/gateway.do?service=notify_verify&';
  8. private $__http_verify_url = 'http://notify.alipay.com/trade/notify_query.do?';
  9. private $service = 'mobile.securitypay.pay';
  10. private $partner;
  11. private $_input_charset = 'UTF-8';
  12. private $sign_type = 'RSA';
  13. private $private_key_path;
  14. private $public_key_path;
  15. private $notify_url;
  16. private $out_trade_no;
  17. private $subject;
  18. private $payment_type = 1;
  19. private $seller_id;
  20. private $total_fee;
  21. private $body;
  22. private $show_url;
  23. private $anti_phishing_key;
  24. private $exter_invoke_ip;
  25. private $key;
  26. private $transport;
  27. private $cacert;
  28. public function __construct()
  29. {
  30. $this->cacert = getcwd() . '\\cacert.pem';
  31. }
  32. public function setPem($cacert)
  33. {
  34. $this->cacert = $cacert;
  35. return $this;
  36. }
  37. /**
  38. * 取得支付链接参数
  39. */
  40. public function getPayPara()
  41. {
  42. $parameter = array(
  43. 'service' => $this->service,
  44. 'partner' => trim($this->partner),
  45. 'payment_type' => $this->payment_type,
  46. 'notify_url' => $this->notify_url,
  47. 'seller_id' => $this->seller_id,
  48. 'out_trade_no' => $this->out_trade_no,
  49. 'subject' => $this->subject,
  50. 'total_fee' => $this->total_fee,
  51. 'body' => $this->body,
  52. 'show_url' => $this->show_url,
  53. 'anti_phishing_key' => $this->anti_phishing_key,
  54. 'exter_invoke_ip' => $this->exter_invoke_ip,
  55. '_input_charset' => trim(strtolower($this->_input_charset))
  56. );
  57. //请求数据日志
  58. event(new PayLog($parameter, new MobileAlipay()));
  59. $para = $this->buildRequestPara($parameter);
  60. return $this->createLinkstringUrlencode($para);
  61. }
  62. /**
  63. * 验证消息是否是支付宝发出的合法消息
  64. */
  65. public function verify()
  66. {
  67. // 判断请求是否为空
  68. if (empty($_POST) && empty($_GET)) {
  69. return false;
  70. }
  71. $data = $_POST ? : $_GET;
  72. // 生成签名结果
  73. $is_sign = $this->getSignVeryfy($data, $data['sign']);
  74. // 获取支付宝远程服务器ATN结果(验证是否是支付宝发来的消息)
  75. $response_txt = 'true';
  76. if (! empty($data['notify_id'])) {
  77. $response_txt = $this->getResponse($data['notify_id']);
  78. }
  79. // 验证
  80. // $response_txt的结果不是true,与服务器设置问题、合作身份者ID、notify_id一分钟失效有关
  81. // isSign的结果不是true,与安全校验码、请求时的参数格式(如:带自定义参数等)、编码格式有关
  82. if (preg_match('/true$/i', $response_txt) && $is_sign) {
  83. return true;
  84. } else {
  85. return false;
  86. }
  87. }
  88. public function setBody($body)
  89. {
  90. $this->body = $body;
  91. return $this;
  92. }
  93. public function setNotifyUrl($notify_url)
  94. {
  95. $this->notify_url = $notify_url;
  96. return $this;
  97. }
  98. public function setOutTradeNo($out_trade_no)
  99. {
  100. $this->out_trade_no = $out_trade_no;
  101. return $this;
  102. }
  103. public function setPartner($partner)
  104. {
  105. $this->partner = $partner;
  106. return $this;
  107. }
  108. public function setPrivateKeyPath($private_key_path)
  109. {
  110. $this->private_key_path = $private_key_path;
  111. return $this;
  112. }
  113. public function setPublicKeyPath($public_key_path)
  114. {
  115. $this->public_key_path = $public_key_path;
  116. return $this;
  117. }
  118. public function setSellerId($seller_id)
  119. {
  120. $this->seller_id = $seller_id;
  121. return $this;
  122. }
  123. public function setSubject($subject)
  124. {
  125. $this->subject = $subject;
  126. return $this;
  127. }
  128. public function setTotalFee($total_fee)
  129. {
  130. $this->total_fee = $total_fee;
  131. return $this;
  132. }
  133. public function setSignType($sign_type)
  134. {
  135. $this->sign_type = $sign_type;
  136. return $this;
  137. }
  138. /**
  139. * 生成要请求给支付宝的参数数组
  140. * @param $para_temp 请求前的参数数组
  141. * @return 要请求的参数数组
  142. */
  143. private function buildRequestPara($para_temp)
  144. {
  145. //除去待签名参数数组中的空值和签名参数
  146. $para_filter = $this->paraFilter($para_temp);
  147. //对待签名参数数组排序
  148. $para_sort = $this->argSort($para_filter);
  149. //生成签名结果
  150. $mysign = $this->buildRequestMysign($para_sort);
  151. //签名结果与签名方式加入请求提交参数组中
  152. $para_sort['sign'] = $mysign;
  153. $para_sort['sign_type'] = strtoupper(trim($this->sign_type));
  154. return $para_sort;
  155. }
  156. /**
  157. * 生成签名结果
  158. * @param $para_sort 已排序要签名的数组
  159. * return 签名结果字符串
  160. */
  161. private function buildRequestMysign($para_sort)
  162. {
  163. //把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
  164. $prestr = $this->createLinkstring($para_sort);
  165. $mysign = '';
  166. switch (strtoupper(trim($this->sign_type))) {
  167. case 'MD5':
  168. $mysign = $this->md5Sign($prestr, $this->key);
  169. break;
  170. case 'RSA':
  171. $mysign = $this->rsaSign($prestr, trim($this->private_key_path));
  172. break;
  173. default:
  174. $mysign = '';
  175. }
  176. return $mysign;
  177. }
  178. /**
  179. * 获取返回时的签名验证结果
  180. * @param $para_temp 通知返回来的参数数组
  181. * @param $sign 返回的签名结果
  182. * @return 签名验证结果
  183. */
  184. function getSignVeryfy($para_temp, $sign)
  185. {
  186. //除去待签名参数数组中的空值和签名参数
  187. $para_filter = $this->paraFilter($para_temp);
  188. //对待签名参数数组排序
  189. $para_sort = $this->argSort($para_filter);
  190. //把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
  191. $prestr = $this->createLinkstring($para_sort);
  192. $is_sgin = false;
  193. switch (strtoupper(trim($this->sign_type))) {
  194. case 'MD5':
  195. $is_sgin = $this->md5Verify($prestr, $sign, $this->key);
  196. break;
  197. case 'RSA':
  198. $is_sgin = $this->rsaVerify($prestr, $this->public_key_path, $sign);
  199. break;
  200. default:
  201. $is_sgin = false;
  202. }
  203. return $is_sgin;
  204. }
  205. /**
  206. * 除去数组中的空值和签名参数
  207. * @param $para 签名参数组
  208. * return 去掉空值与签名参数后的新签名参数组
  209. */
  210. private function paraFilter($para)
  211. {
  212. $para_filter = array();
  213. while ((list ($key, $val) = each($para)) == true) {
  214. if ($key == 'sign' || $key == 'sign_type' || $val == '') {
  215. continue;
  216. } else {
  217. $para_filter[$key] = $para[$key];
  218. }
  219. }
  220. return $para_filter;
  221. }
  222. /**
  223. * 对数组排序
  224. * @param $para 排序前的数组
  225. * return 排序后的数组
  226. */
  227. private function argSort($para)
  228. {
  229. ksort($para);
  230. reset($para);
  231. return $para;
  232. }
  233. /**
  234. * RSA验签
  235. * @param $data 待签名数据
  236. * @param $ali_public_key_path 支付宝的公钥文件路径
  237. * @param $sign 要校对的的签名结果
  238. * return 验证结果
  239. */
  240. private function rsaVerify($data, $public_key_path, $sign)
  241. {
  242. $pubKey = file_get_contents($public_key_path);
  243. $res = openssl_get_publickey($pubKey);
  244. $result = (bool) openssl_verify($data, base64_decode($sign), $res);
  245. openssl_free_key($res);
  246. return $result;
  247. }
  248. /**
  249. * RSA签名
  250. * @param $data 待签名数据
  251. * @param $private_key_path 商户私钥文件路径
  252. * return 签名结果
  253. */
  254. private function rsaSign($data, $private_key_path)
  255. {
  256. $priKey = file_get_contents($private_key_path);
  257. $res = openssl_get_privatekey($priKey);
  258. openssl_sign($data, $sign, $res);
  259. openssl_free_key($res);
  260. //base64编码
  261. $sign = base64_encode($sign);
  262. return $sign;
  263. }
  264. /**
  265. * 把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
  266. * @param $para 需要拼接的数组
  267. * return 拼接完成以后的字符串
  268. */
  269. private function createLinkstring($para)
  270. {
  271. $arg = '';
  272. while ((list ($key, $val) = each($para)) == true) {
  273. $arg .= $key . '=' . $val . '&';
  274. }
  275. //去掉最后一个&字符
  276. $arg = substr($arg, 0, count($arg) - 2);
  277. //如果存在转义字符,那么去掉转义
  278. if (get_magic_quotes_gpc()) {
  279. $arg = stripslashes($arg);
  280. }
  281. return $arg;
  282. }
  283. /**
  284. * 把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串,并对字符串做urlencode编码
  285. * @param $para 需要拼接的数组
  286. * return 拼接完成以后的字符串
  287. */
  288. private function createLinkstringUrlencode($para)
  289. {
  290. $arg = '';
  291. while ((list ($key, $val) = each($para)) == true) {
  292. $arg .= $key . '=' . urlencode($val) . '&';
  293. }
  294. //去掉最后一个&字符
  295. $arg = substr($arg, 0, count($arg) - 2);
  296. //如果存在转义字符,那么去掉转义
  297. if (get_magic_quotes_gpc()) {
  298. $arg = stripslashes($arg);
  299. }
  300. return $arg;
  301. }
  302. /**
  303. * 获取远程服务器ATN结果,验证返回URL
  304. * @param $notify_id 通知校验ID
  305. * @return 服务器ATN结果
  306. * 验证结果集:
  307. * invalid命令参数不对 出现这个错误,请检测返回处理中partner和key是否为空
  308. * true 返回正确信息
  309. * false 请检查防火墙或者是服务器阻止端口问题以及验证时间是否超过一分钟
  310. */
  311. private function getResponse($notify_id)
  312. {
  313. $transport = strtolower(trim($this->transport));
  314. $partner = trim($this->partner);
  315. $veryfy_url = '';
  316. if ($transport == 'https') {
  317. $veryfy_url = $this->__https_verify_url;
  318. } else {
  319. $veryfy_url = $this->__http_verify_url;
  320. }
  321. $veryfy_url = $veryfy_url . 'partner=' . $partner . '&notify_id=' . $notify_id;
  322. $response_txt = $this->getHttpResponseGET($veryfy_url, $this->cacert);
  323. return $response_txt;
  324. }
  325. /**
  326. * 远程获取数据,GET模式
  327. * 注意:
  328. * 1.使用Crul需要修改服务器中php.ini文件的设置,找到php_curl.dll去掉前面的";"就行了
  329. * 2.文件夹中cacert.pem是SSL证书请保证其路径有效,目前默认路径是:getcwd().'\\cacert.pem'
  330. * @param $url 指定URL完整路径地址
  331. * @param $cacert_url 指定当前工作目录绝对路径
  332. * return 远程输出的数据
  333. */
  334. private function getHttpResponseGET($url, $cacert_url)
  335. {
  336. $curl = curl_init($url);
  337. curl_setopt($curl, CURLOPT_HEADER, 0); // 过滤HTTP头
  338. curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 显示输出结果
  339. curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true); //SSL证书认证
  340. curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2); //严格认证
  341. curl_setopt($curl, CURLOPT_CAINFO, $cacert_url); //证书地址
  342. $responseText = curl_exec($curl);
  343. //var_dump( curl_error($curl) );//如果执行curl过程中出现异常,可打开此开关,以便查看异常内容
  344. curl_close($curl);
  345. return $responseText;
  346. }
  347. }