BasicWePay.php 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | WeChatDeveloper
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2022 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://think.ctolog.com
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( https://mit-license.org )
  10. // +----------------------------------------------------------------------
  11. // | github开源项目:https://github.com/zoujingli/WeChatDeveloper
  12. // +----------------------------------------------------------------------
  13. namespace WeChat\Contracts;
  14. use WeChat\Exceptions\InvalidArgumentException;
  15. use WeChat\Exceptions\InvalidResponseException;
  16. /**
  17. * 微信支付基础类
  18. * Class BasicPay
  19. * @package WeChat\Contracts
  20. */
  21. class BasicWePay
  22. {
  23. /**
  24. * 商户配置
  25. * @var DataArray
  26. */
  27. protected $config;
  28. /**
  29. * 当前请求数据
  30. * @var DataArray
  31. */
  32. protected $params;
  33. /**
  34. * 静态缓存
  35. * @var static
  36. */
  37. protected static $cache;
  38. /**
  39. * WeChat constructor.
  40. * @param array $options
  41. */
  42. public function __construct(array $options)
  43. {
  44. if (empty($options['appid'])) {
  45. throw new InvalidArgumentException("Missing Config -- [appid]");
  46. }
  47. if (empty($options['mch_id'])) {
  48. throw new InvalidArgumentException("Missing Config -- [mch_id]");
  49. }
  50. if (empty($options['mch_key'])) {
  51. throw new InvalidArgumentException("Missing Config -- [mch_key]");
  52. }
  53. if (!empty($options['cache_path'])) {
  54. Tools::$cache_path = $options['cache_path'];
  55. }
  56. $this->config = new DataArray($options);
  57. // 商户基础参数
  58. $this->params = new DataArray([
  59. 'appid' => $this->config->get('appid'),
  60. 'mch_id' => $this->config->get('mch_id'),
  61. 'nonce_str' => Tools::createNoncestr(),
  62. ]);
  63. // 商户参数支持
  64. if ($this->config->get('sub_appid')) {
  65. $this->params->set('sub_appid', $this->config->get('sub_appid'));
  66. }
  67. if ($this->config->get('sub_mch_id')) {
  68. $this->params->set('sub_mch_id', $this->config->get('sub_mch_id'));
  69. }
  70. }
  71. /**
  72. * 静态创建对象
  73. * @param array $config
  74. * @return static
  75. */
  76. public static function instance(array $config)
  77. {
  78. $key = md5(get_called_class() . serialize($config));
  79. if (isset(self::$cache[$key])) return self::$cache[$key];
  80. return self::$cache[$key] = new static($config);
  81. }
  82. /**
  83. * 获取微信支付通知
  84. * @return array
  85. * @throws InvalidResponseException
  86. */
  87. public function getNotify()
  88. {
  89. $data = Tools::xml2arr(file_get_contents('php://input'));
  90. if (isset($data['sign']) && $this->getPaySign($data) === $data['sign']) {
  91. return $data;
  92. }
  93. throw new InvalidResponseException('Invalid Notify.', '0');
  94. }
  95. /**
  96. * 获取微信支付通知回复内容
  97. * @return string
  98. */
  99. public function getNotifySuccessReply()
  100. {
  101. return Tools::arr2xml(['return_code' => 'SUCCESS', 'return_msg' => 'OK']);
  102. }
  103. /**
  104. * 生成支付签名
  105. * @param array $data 参与签名的数据
  106. * @param string $signType 参与签名的类型
  107. * @param string $buff 参与签名字符串前缀
  108. * @return string
  109. */
  110. public function getPaySign(array $data, $signType = 'MD5', $buff = '')
  111. {
  112. ksort($data);
  113. if (isset($data['sign'])) unset($data['sign']);
  114. foreach ($data as $k => $v) $buff .= "{$k}={$v}&";
  115. $buff .= ("key=" . $this->config->get('mch_key'));
  116. if (strtoupper($signType) === 'MD5') {
  117. return strtoupper(md5($buff));
  118. }
  119. return strtoupper(hash_hmac('SHA256', $buff, $this->config->get('mch_key')));
  120. }
  121. /**
  122. * 转换短链接
  123. * @param string $longUrl 需要转换的URL,签名用原串,传输需URLencode
  124. * @return array
  125. * @throws InvalidResponseException
  126. * @throws \WeChat\Exceptions\LocalCacheException
  127. */
  128. public function shortUrl($longUrl)
  129. {
  130. $url = 'https://api.mch.weixin.qq.com/tools/shorturl';
  131. return $this->callPostApi($url, ['long_url' => $longUrl]);
  132. }
  133. /**
  134. * 数组直接转xml数据输出
  135. * @param array $data
  136. * @param bool $isReturn
  137. * @return string
  138. */
  139. public function toXml(array $data, $isReturn = false)
  140. {
  141. $xml = Tools::arr2xml($data);
  142. if ($isReturn) {
  143. return $xml;
  144. }
  145. echo $xml;
  146. }
  147. /**
  148. * 以Post请求接口
  149. * @param string $url 请求
  150. * @param array $data 接口参数
  151. * @param bool $isCert 是否需要使用双向证书
  152. * @param string $signType 数据签名类型 MD5|SHA256
  153. * @param bool $needSignType 是否需要传签名类型参数
  154. * @return array
  155. * @throws InvalidResponseException
  156. * @throws \WeChat\Exceptions\LocalCacheException
  157. */
  158. protected function callPostApi($url, array $data, $isCert = false, $signType = 'HMAC-SHA256', $needSignType = true, $needNonceStr = true)
  159. {
  160. $option = [];
  161. if ($isCert) {
  162. $option['ssl_p12'] = $this->config->get('ssl_p12');
  163. $option['ssl_cer'] = $this->config->get('ssl_cer');
  164. $option['ssl_key'] = $this->config->get('ssl_key');
  165. if (is_string($option['ssl_p12']) && file_exists($option['ssl_p12'])) {
  166. $content = file_get_contents($option['ssl_p12']);
  167. if (openssl_pkcs12_read($content, $certs, $this->config->get('mch_id'))) {
  168. $option['ssl_key'] = Tools::pushFile(md5($certs['pkey']) . '.pem', $certs['pkey']);
  169. $option['ssl_cer'] = Tools::pushFile(md5($certs['cert']) . '.pem', $certs['cert']);
  170. } else throw new InvalidArgumentException("P12 certificate does not match MCH_ID --- ssl_p12");
  171. }
  172. if (empty($option['ssl_cer']) || !file_exists($option['ssl_cer'])) {
  173. throw new InvalidArgumentException("Missing Config -- ssl_cer", '0');
  174. }
  175. if (empty($option['ssl_key']) || !file_exists($option['ssl_key'])) {
  176. throw new InvalidArgumentException("Missing Config -- ssl_key", '0');
  177. }
  178. }
  179. $params = $this->params->merge($data);
  180. if (!$needNonceStr) unset($params['nonce_str']);
  181. if ($needSignType) $params['sign_type'] = strtoupper($signType);
  182. $params['sign'] = $this->getPaySign($params, $signType);
  183. $result = Tools::xml2arr(Tools::post($url, Tools::arr2xml($params), $option));
  184. if ($result['return_code'] !== 'SUCCESS') {
  185. throw new InvalidResponseException($result['return_msg'], '0');
  186. }
  187. return $result;
  188. }
  189. }