123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 |
- <?php
- // +----------------------------------------------------------------------
- // | WeChatDeveloper
- // +----------------------------------------------------------------------
- // | 版权所有 2014~2023 ThinkAdmin [ thinkadmin.top ]
- // +----------------------------------------------------------------------
- // | 官方网站: https://thinkadmin.top
- // +----------------------------------------------------------------------
- // | 开源协议 ( https://mit-license.org )
- // | 免责声明 ( https://thinkadmin.top/disclaimer )
- // +----------------------------------------------------------------------
- // | gitee 代码仓库:https://gitee.com/zoujingli/WeChatDeveloper
- // | github 代码仓库:https://github.com/zoujingli/WeChatDeveloper
- // +----------------------------------------------------------------------
- namespace WePayV3\Contracts;
- use WeChat\Contracts\Tools;
- use WeChat\Exceptions\InvalidArgumentException;
- use WeChat\Exceptions\InvalidDecryptException;
- use WeChat\Exceptions\InvalidResponseException;
- 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 bool
- */
- protected $autoCert = true;
- /**
- * 配置参数
- * @var array
- */
- protected $config = [
- 'appid' => '', // 微信绑定APPID,需配置
- 'mch_id' => '', // 微信商户编号,需要配置
- 'mch_v3_key' => '', // 微信商户密钥,需要配置
- 'cert_serial' => '', // 商户证书序号,无需配置
- 'cert_public' => '', // 商户公钥内容,需要配置
- 'cert_private' => '', // 商户密钥内容,需要配置
- 'mp_cert_serial' => '', // 平台证书序号,无需配置
- 'mp_cert_content' => '', // 平台证书内容,无需配置
- ];
- /**
- * BasicWePayV3 constructor.
- * @param array $options [mch_id, mch_v3_key, cert_public, cert_private]
- * @throws \WeChat\Exceptions\InvalidResponseException
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- 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_public'])) {
- throw new InvalidArgumentException("Missing Config -- [cert_public]");
- }
- if (empty($options['cert_private'])) {
- throw new InvalidArgumentException("Missing Config -- [cert_private]");
- }
- 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'];
- if (empty($options['cert_serial'])) {
- $this->config['cert_serial'] = openssl_x509_parse($this->config['cert_public'], true)['serialNumberHex'];
- } else {
- $this->config['cert_serial'] = $options['cert_serial'];
- }
- if (empty($this->config['cert_serial'])) {
- throw new InvalidArgumentException('Failed to parse certificate public key');
- }
- if (!empty($options['cache_path'])) {
- Tools::$cache_path = $options['cache_path'];
- }
- // 自动配置平台证书
- if ($this->autoCert) {
- $this->_autoCert();
- }
- // 服务商参数支持
- // if (!empty($options['sp_appid'])) {
- // $this->config['sp_appid'] = $options['sp_appid'];
- // }
- // if (!empty($options['sp_mchid'])) {
- // $this->config['sp_mchid'] = $options['sp_mchid'];
- // }
- // if (!empty($options['sub_appid'])) {
- // $this->config['sub_appid'] = $options['sub_appid'];
- // }
- // if (!empty($options['sub_mch_id'])) {
- // $this->config['sub_mch_id'] = $options['sub_mch_id'];
- // }
- }
- /**
- * 静态创建对象
- * @param array $config
- * @return static
- * @throws \WeChat\Exceptions\InvalidResponseException
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- 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 boolean $verify 是否验证
- * @param boolean $isjson 返回JSON
- * @return array|string
- * @throws \WeChat\Exceptions\InvalidResponseException
- */
- public function doRequest($method, $pathinfo, $jsondata = '', $verify = false, $isjson = true)
- {
- 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)
- );
- $location = (preg_match('|^https?://|', $pathinfo) ? '' : $this->base) . $pathinfo;
- list($header, $content) = $this->_doRequestCurl($method, $location, [
- 'data' => $jsondata, 'header' => [
- 'Accept: application/json',
- 'Content-Type: application/json',
- 'User-Agent: https://thinkadmin.top',
- "Authorization: WECHATPAY2-SHA256-RSA2048 {$token}",
- "Wechatpay-Serial: {$this->config['mp_cert_serial']}"
- ],
- ]);
- 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 {
- if (empty($headers)) {
- return $isjson ? json_decode($content, true) : $content;
- }
- $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 $isjson ? json_decode($content, true) : $content;
- }
- /**
- * 通过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 \WeChat\Exceptions\InvalidResponseException
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- protected function signVerify($data, $sign, $serial)
- {
- $cert = $this->_getCert($serial);
- return @openssl_verify($data, base64_decode($sign), openssl_x509_read($cert), 'sha256WithRSAEncryption');
- }
- /**
- * 获取平台证书
- * @param string $serial
- * @return string
- * @throws \WeChat\Exceptions\InvalidResponseException
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- protected function _getCert($serial = '')
- {
- $certs = $this->tmpFile("{$this->config['mch_id']}_certs");
- if (empty($certs) || empty($certs[$serial]['content'])) {
- Cert::instance($this->config)->download();
- $certs = $this->tmpFile("{$this->config['mch_id']}_certs");
- }
- if (empty($certs[$serial]['content']) || $certs[$serial]['expire'] < time()) {
- throw new InvalidResponseException("读取平台证书失败!");
- } else {
- return $certs[$serial]['content'];
- }
- }
- /**
- * 自动配置平台证书
- * @return void
- * @throws \WeChat\Exceptions\InvalidResponseException
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- protected function _autoCert()
- {
- $certs = $this->tmpFile("{$this->config['mch_id']}_certs");
- if (is_array($certs)) foreach ($certs as $k => $v) {
- if ($v['expire'] < time()) unset($certs[$k]);
- }
- if (empty($certs)) {
- Cert::instance($this->config)->download();
- $certs = $this->tmpFile("{$this->config['mch_id']}_certs");
- }
- if (empty($certs) || !is_array($certs)) {
- throw new InvalidResponseException("读取平台证书失败!");
- }
- foreach ($certs as $k => $v) if ($v['expire'] > time() + 10) {
- $this->config['mp_cert_serial'] = $k;
- $this->config['mp_cert_content'] = $v['content'];
- break;
- }
- if (empty($this->config['mp_cert_serial']) || empty($this->config['mp_cert_content'])) {
- throw new InvalidResponseException("自动配置平台证书失败!");
- }
- }
- /**
- * 写入或读取临时文件
- * @param string $name
- * @param null|array|string $content
- * @param integer $expire
- * @return array|string
- * @throws \WeChat\Exceptions\LocalCacheException
- */
- protected function tmpFile($name, $content = null, $expire = 7200)
- {
- if (is_null($content)) {
- $text = Tools::getCache($name);
- if (empty($text)) return '';
- $json = json_decode(Tools::getCache($name) ?: '', true);
- return isset($json[0]) ? $json[0] : '';
- } else {
- return Tools::setCache($name, json_encode([$content], JSON_UNESCAPED_UNICODE), $expire);
- }
- }
- /**
- * RSA加密处理-平台证书
- * @param string $string
- * @return string
- * @throws \WeChat\Exceptions\InvalidDecryptException
- */
- protected function rsaEncode($string)
- {
- $publicKey = $this->config['mp_cert_content'];
- if (openssl_public_encrypt($string, $encrypted, $publicKey, OPENSSL_PKCS1_OAEP_PADDING)) {
- return base64_encode($encrypted);
- } else {
- throw new InvalidDecryptException('Rsa Encrypt Error.');
- }
- }
- }
|