Common.php 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. <?php
  2. class Common
  3. {
  4. // 商户号
  5. protected $sellerMid = '6888803045944';
  6. // 公钥文件
  7. // protected $publicKeyPath = '/www/wwwroot/xiang.qcyjsh.com/application/common/library/shande/cert/6888803045944.cer';
  8. protected $publicKeyPath = '/www/wwwroot/xiang_admin.com/application/common/library/shande/cert/sand.cer';
  9. // 私钥文件
  10. protected $privateKeyPath = '/www/wwwroot/xiang_admin.qcyjsh.com/application/common/library/shande/cert/6888803045944.pfx';
  11. // 私钥证书密码
  12. protected $privateKeyPwd = '980801';
  13. // 接口地址
  14. protected $apiUrl = 'https://cashier.sandpay.com.cn';
  15. // 产品id https://open.sandpay.com.cn/product/detail/43984//
  16. protected $productId = '';
  17. // 接入类型 1-普通商户接入 2-平台商户接入 3-核心企业商户接入
  18. protected $accessType = '1';
  19. // 渠道类型 07-互联网 08-移动端
  20. protected $channelType = '07';
  21. // 平台ID accessType为2时必填,在担保支付模式下填写核心商户号
  22. protected $plMid = '';
  23. // 请求报文体
  24. public $body;
  25. /*
  26. |--------------------------------------------------------------------------
  27. | step1.组装参数
  28. |--------------------------------------------------------------------------
  29. */
  30. // 参数
  31. protected function postData($method)
  32. {
  33. $data = array(
  34. 'head' => array(
  35. 'version' => '1.0',
  36. 'method' => $method,
  37. 'productId' => $this->productId,
  38. 'accessType' => $this->accessType,
  39. 'mid' => $this->sellerMid,
  40. 'plMid' => $this->plMid,
  41. 'channelType' => $this->channelType,
  42. 'reqTime' => date('YmdHis', time()),
  43. ),
  44. 'body' => $this->body,
  45. );
  46. $postData = array(
  47. 'charset' => 'utf-8',
  48. 'signType' => '01',
  49. 'data' => json_encode($data),
  50. 'sign' => $this->sign($data),
  51. );
  52. file_put_contents("shandevalues.txt", json_encode($postData,true) . "\n" , FILE_APPEND);
  53. return $postData;
  54. }
  55. protected function newPostData($method,$goods_name,$meta_option,$pay_extra)
  56. {
  57. $data = [
  58. 'version' => '10',
  59. 'mer_no' => $this->sellerMid,
  60. 'mer_key' => $this->privateKeyPwd,
  61. 'create_time' => date('YmdHis'),
  62. 'order_amt' => 0.01,
  63. 'notify_url' => '',
  64. 'return_url' => '',
  65. 'create_ip' => request()->ip(),
  66. 'pay_extra' => $pay_extra,
  67. 'accsplit_flag' => 'NO',
  68. 'sign_type' => 'MD5',
  69. 'store_id' => '000000',
  70. 'expire_time' => date('YmdHis',time()+600),
  71. 'goods_name' => $goods_name,
  72. 'sign' => '$goods_name',
  73. 'jump_scheme' => 'sandcash://scpay',
  74. 'meta_option' => $meta_option,
  75. ];
  76. }
  77. // 参数映射 继承类需要完善这个方法
  78. protected function apiMap()
  79. {
  80. return array();
  81. }
  82. /*
  83. |--------------------------------------------------------------------------
  84. | step2. 请求
  85. |--------------------------------------------------------------------------
  86. */
  87. // curl请求接口
  88. public function request($apiName)
  89. {
  90. try {
  91. $apiMap = $this->apiMap();
  92. if (!isset($apiMap[$apiName])) {
  93. throw new \Exception('接口名错误');
  94. }
  95. $postData = $this->postData($apiMap[$apiName]['method']);
  96. $url = $this->apiUrl . $apiMap[$apiName]['url'];
  97. $ret = $this->httpPost($url, $postData);
  98. $retAry = $this->parseResult($ret); // 格式解析
  99. $verify = $this->verify($retAry['data'], $retAry['sign']); // 验签
  100. if (!$verify) {
  101. throw new \Exception('返回数据验签失败');
  102. }
  103. return $retAry;
  104. } catch (\Exception $e) {
  105. return $e->getMessage();
  106. }
  107. }
  108. // curl. 发送请求
  109. protected function httpPost($url, $params)
  110. {
  111. if (empty($url) || empty($params)) {
  112. throw new \Exception('请求参数错误');
  113. }
  114. $params = http_build_query($params);
  115. try {
  116. $ch = curl_init();
  117. curl_setopt($ch, CURLOPT_URL, $url);
  118. curl_setopt($ch, CURLOPT_POST, 1);
  119. curl_setopt($ch, CURLOPT_POSTFIELDS, $params);
  120. curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type: application/x-www-form-urlencoded'));
  121. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  122. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
  123. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
  124. $data = curl_exec($ch);
  125. $err = curl_error($ch);
  126. $errno = curl_errno($ch);
  127. if ($errno) {
  128. $msg = 'curl errInfo: ' . $err . ' curl errNo: ' . $errno;
  129. throw new \Exception($msg);
  130. }
  131. curl_close($ch);
  132. return $data;
  133. } catch (\Exception $e) {
  134. if ($ch) curl_close($ch);
  135. throw $e;
  136. }
  137. }
  138. // curl.解析返回数据
  139. protected function parseResult($result)
  140. {
  141. $arr = array();
  142. $response = urldecode($result);
  143. $arrStr = explode('&', $response);
  144. foreach ($arrStr as $str) {
  145. $p = strpos($str, "=");
  146. $key = substr($str, 0, $p);
  147. $value = substr($str, $p + 1);
  148. $arr[$key] = $value;
  149. }
  150. return $arr;
  151. }
  152. // 表单请求接口
  153. public function form($apiName)
  154. {
  155. $apiMap = $this->apiMap();
  156. if (!isset($apiMap[$apiName])) {
  157. throw new \Exception('接口名错误');
  158. }
  159. $postData = $this->postData($apiMap[$apiName]['method']);
  160. $url = $this->apiUrl . $apiMap[$apiName]['url'];
  161. $form = '<form action="' . $url . '" method="post">';
  162. foreach ($postData as $k => $v) {
  163. $form .= "{$k} <p><input type='text' name='{$k}' value='{$v}'></p>";
  164. }
  165. $form .= '<input type="submit" value="提交"></form>';
  166. return $form;
  167. }
  168. /*
  169. |--------------------------------------------------------------------------
  170. | step3.签名 + 验签
  171. |--------------------------------------------------------------------------
  172. */
  173. // 公钥
  174. private function publicKey()
  175. {
  176. try {
  177. $file = file_get_contents($this->publicKeyPath);
  178. if (!$file) {
  179. throw new \Exception('getPublicKey::file_get_contents ERROR');
  180. }
  181. $cert = chunk_split(base64_encode($file), 64, "\n");
  182. $cert = "-----BEGIN CERTIFICATE-----\n" . $cert . "-----END CERTIFICATE-----\n";
  183. $res = openssl_pkey_get_public($cert);
  184. $detail = openssl_pkey_get_details($res);
  185. openssl_free_key($res);
  186. if (!$detail) {
  187. throw new \Exception('getPublicKey::openssl_pkey_get_details ERROR');
  188. }
  189. return $detail['key'];
  190. } catch (\Exception $e) {
  191. throw $e;
  192. }
  193. }
  194. // 私钥
  195. private function privateKey()
  196. {
  197. try {
  198. $file = file_get_contents($this->privateKeyPath);
  199. if (!$file) {
  200. throw new \Exception('getPrivateKey::file_get_contents');
  201. }
  202. if (!openssl_pkcs12_read($file, $cert, $this->privateKeyPwd)) {
  203. throw new \Exception('getPrivateKey::openssl_pkcs12_read ERROR');
  204. }
  205. return $cert['pkey'];
  206. } catch (\Exception $e) {
  207. throw $e;
  208. }
  209. }
  210. // 私钥加签
  211. protected function sign($plainText)
  212. {
  213. $plainText = json_encode($plainText);
  214. try {
  215. $resource = openssl_pkey_get_private($this->privateKey());
  216. $result = openssl_sign($plainText, $sign, $resource);
  217. openssl_free_key($resource);
  218. if (!$result) throw new \Exception('sign error');
  219. return base64_encode($sign);
  220. } catch (\Exception $e) {
  221. throw $e;
  222. }
  223. }
  224. // 公钥验签
  225. public function verify($plainText, $sign)
  226. {
  227. $resource = openssl_pkey_get_public($this->publicKey());
  228. $result = openssl_verify($plainText, base64_decode($sign), $resource);
  229. openssl_free_key($resource);
  230. if (!$result) {
  231. throw new \Exception('签名验证未通过,plainText:|' . $plainText . '|。sign:' . $sign);
  232. }
  233. return $result;
  234. }
  235. }