| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235 |
- <?php
- // +----------------------------------------------------------------------
- // | WeChatDeveloper
- // +----------------------------------------------------------------------
- // | 版权所有 2014~2020 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
- // +----------------------------------------------------------------------
- // | 官方网站: http://think.ctolog.com
- // +----------------------------------------------------------------------
- // | 开源协议 ( https://mit-license.org )
- // +----------------------------------------------------------------------
- // | github开源项目:https://github.com/zoujingli/WeChatDeveloper
- // +----------------------------------------------------------------------
- namespace WePayV3\Contracts;
- use WeChat\Contracts\Tools;
- use WeChat\Exceptions\InvalidArgumentException;
- use WeChat\Exceptions\InvalidResponseException;
- use WeChat\Exceptions\LocalCacheException;
- use WePayV3\Cert;
- /**
- * 微信支付基础类
- * Class BasicWePay
- * @package WePayV3
- */
- abstract class BasicWePay
- {
- /**
- * 接口基础地址
- * @var string
- */
- protected $base = 'https://api.mch.weixin.qq.com';
- /**
- * 实例对象静态缓存
- * @var array
- */
- static $cache = [];
- /**
- * 配置参数
- * @var array
- */
- protected $config = [
- 'appid' => '', // 微信绑定APPID,需配置
- 'mch_id' => '', // 微信商户编号,需要配置
- 'mch_v3_key' => '', // 微信商户密钥,需要配置
- 'cert_serial' => '', // 商户证书序号,无需配置
- 'cert_public' => '', // 商户公钥内容,需要配置
- 'cert_private' => '', // 商户密钥内容,需要配置
- ];
- /**
- * BasicWePayV3 constructor.
- * @param array $options [mch_id, mch_v3_key, cert_public, cert_private]
- */
- public function __construct(array $options = [])
- {
- if (empty($options['mch_id'])) {
- throw new InvalidArgumentException("Missing Config -- [mch_id]");
- }
- if (empty($options['mch_v3_key'])) {
- throw new InvalidArgumentException("Missing Config -- [mch_v3_key]");
- }
- if (empty($options['cert_private'])) {
- throw new InvalidArgumentException("Missing Config -- [cert_private]");
- }
- if (empty($options['cert_public'])) {
- throw new InvalidArgumentException("Missing Config -- [cert_public]");
- }
- if (stripos($options['cert_public'], '-----BEGIN CERTIFICATE-----') === false) {
- if (file_exists($options['cert_public'])) {
- $options['cert_public'] = file_get_contents($options['cert_public']);
- } else {
- throw new InvalidArgumentException("File Non-Existent -- [cert_public]");
- }
- }
- if (stripos($options['cert_private'], '-----BEGIN PRIVATE KEY-----') === false) {
- if (file_exists($options['cert_private'])) {
- $options['cert_private'] = file_get_contents($options['cert_private']);
- } else {
- throw new InvalidArgumentException("File Non-Existent -- [cert_private]");
- }
- }
- $this->config['appid'] = isset($options['appid']) ? $options['appid'] : '';
- $this->config['mch_id'] = $options['mch_id'];
- $this->config['mch_v3_key'] = $options['mch_v3_key'];
- $this->config['cert_public'] = $options['cert_public'];
- $this->config['cert_private'] = $options['cert_private'];
- $this->config['cert_serial'] = openssl_x509_parse($this->config['cert_public'])['serialNumberHex'];
- if (empty($this->config['cert_serial'])) {
- throw new InvalidArgumentException("Failed to parse certificate public key");
- }
- }
- /**
- * 静态创建对象
- * @param array $config
- * @return static
- */
- public static function instance($config)
- {
- $key = md5(get_called_class() . serialize($config));
- if (isset(self::$cache[$key])) return self::$cache[$key];
- return self::$cache[$key] = new static($config);
- }
- /**
- * 模拟发起请求
- * @param string $method 请求访问
- * @param string $pathinfo 请求路由
- * @param string $jsondata 请求数据
- * @param bool $verify 是否验证
- * @return array
- * @throws InvalidResponseException
- */
- public function doRequest($method, $pathinfo, $jsondata = '', $verify = false)
- {
- list($time, $nonce) = [time(), uniqid() . rand(1000, 9999)];
- $signstr = join("\n", [$method, $pathinfo, $time, $nonce, $jsondata, '']);
- // 生成数据签名TOKEN
- $token = sprintf('mchid="%s",nonce_str="%s",timestamp="%d",serial_no="%s",signature="%s"',
- $this->config['mch_id'], $nonce, $time, $this->config['cert_serial'], $this->signBuild($signstr)
- );
- list($header, $content) = $this->_doRequestCurl($method, $this->base . $pathinfo, [
- 'data' => $jsondata, 'header' => [
- "Accept: application/json", "Content-Type: application/json",
- 'User-Agent: https://thinkadmin.top', "Authorization: WECHATPAY2-SHA256-RSA2048 {$token}",
- ],
- ]);
- if ($verify) {
- $headers = [];
- foreach (explode("\n", $header) as $line) {
- if (stripos($line, 'Wechatpay') !== false) {
- list($name, $value) = explode(':', $line);
- list(, $keys) = explode('wechatpay-', strtolower($name));
- $headers[$keys] = trim($value);
- }
- }
- try {
- $string = join("\n", [$headers['timestamp'], $headers['nonce'], $content, '']);
- if (!$this->signVerify($string, $headers['signature'], $headers['serial'])) {
- throw new InvalidResponseException("验证响应签名失败");
- }
- } catch (\Exception $exception) {
- throw new InvalidResponseException($exception->getMessage(), $exception->getCode());
- }
- }
- return json_decode($content, true);
- }
- /**
- * 通过CURL模拟网络请求
- * @param string $method 请求方法
- * @param string $location 请求方法
- * @param array $options 请求参数 [data, header]
- * @return array [header,content]
- */
- private function _doRequestCurl($method, $location, $options = [])
- {
- $curl = curl_init();
- // POST数据设置
- if (strtolower($method) === 'post') {
- curl_setopt($curl, CURLOPT_POST, true);
- curl_setopt($curl, CURLOPT_POSTFIELDS, $options['data']);
- }
- // CURL头信息设置
- if (!empty($options['header'])) {
- curl_setopt($curl, CURLOPT_HTTPHEADER, $options['header']);
- }
- curl_setopt($curl, CURLOPT_URL, $location);
- curl_setopt($curl, CURLOPT_HEADER, true);
- curl_setopt($curl, CURLOPT_TIMEOUT, 60);
- curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
- $content = curl_exec($curl);
- $headerSize = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
- curl_close($curl);
- return [substr($content, 0, $headerSize), substr($content, $headerSize)];
- }
- /**
- * 生成数据签名
- * @param string $data 签名内容
- * @return string
- */
- protected function signBuild($data)
- {
- $pkeyid = openssl_pkey_get_private($this->config['cert_private']);
- openssl_sign($data, $signature, $pkeyid, 'sha256WithRSAEncryption');
- return base64_encode($signature);
- }
- /**
- * 验证内容签名
- * @param string $data 签名内容
- * @param string $sign 原签名值
- * @param string $serial 证书序号
- * @return int
- * @throws InvalidResponseException
- * @throws LocalCacheException
- */
- protected function signVerify($data, $sign, $serial = '')
- {
- $cert = $this->tmpFile($serial);
- if (empty($cert)) {
- Cert::instance($this->config)->download();
- $cert = $this->tmpFile($serial);
- }
- return @openssl_verify($data, base64_decode($sign), openssl_x509_read($cert), 'sha256WithRSAEncryption');
- }
- /**
- * 写入或读取临时文件
- * @param string $name
- * @param null|string $content
- * @return string
- * @throws LocalCacheException
- */
- protected function tmpFile($name, $content = null)
- {
- if (is_null($content)) {
- return base64_decode(Tools::getCache($name) ?: '');
- } else {
- return Tools::setCache($name, base64_encode($content), 7200);
- }
- }
- }
|