AddPayloadSignaturePlugin.php 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. <?php
  2. declare(strict_types=1);
  3. namespace Yansongda\Pay\Plugin\Wechat\V3;
  4. use Closure;
  5. use Throwable;
  6. use Yansongda\Artful\Contract\PluginInterface;
  7. use Yansongda\Artful\Exception\ContainerException;
  8. use Yansongda\Artful\Exception\InvalidConfigException;
  9. use Yansongda\Artful\Exception\InvalidParamsException;
  10. use Yansongda\Artful\Exception\ServiceNotFoundException;
  11. use Yansongda\Artful\Logger;
  12. use Yansongda\Artful\Rocket;
  13. use Yansongda\Pay\Exception\Exception;
  14. use Yansongda\Supports\Collection;
  15. use Yansongda\Supports\Str;
  16. use function Yansongda\Pay\get_provider_config;
  17. use function Yansongda\Pay\get_public_cert;
  18. use function Yansongda\Pay\get_wechat_body;
  19. use function Yansongda\Pay\get_wechat_method;
  20. use function Yansongda\Pay\get_wechat_sign;
  21. use function Yansongda\Pay\get_wechat_url;
  22. class AddPayloadSignaturePlugin implements PluginInterface
  23. {
  24. /**
  25. * @throws ContainerException
  26. * @throws InvalidConfigException
  27. * @throws InvalidParamsException
  28. * @throws ServiceNotFoundException
  29. * @throws Throwable 随机数生成失败
  30. */
  31. public function assembly(Rocket $rocket, Closure $next): Rocket
  32. {
  33. Logger::debug('[Wechat][V3][AddPayloadSignaturePlugin] 插件开始装载', ['rocket' => $rocket]);
  34. $config = get_provider_config('wechat', $rocket->getParams());
  35. $payload = $rocket->getPayload();
  36. $timestamp = time();
  37. $random = Str::random(32);
  38. $signContent = $this->getSignatureContent($config, $payload, $timestamp, $random);
  39. $signature = $this->getSignature($config, $timestamp, $random, $signContent);
  40. $rocket->mergePayload(['_authorization' => $signature]);
  41. Logger::info('[Wechat][V3][AddPayloadSignaturePlugin] 插件装载完毕', ['rocket' => $rocket]);
  42. return $next($rocket);
  43. }
  44. /**
  45. * @throws InvalidParamsException
  46. */
  47. protected function getSignatureContent(array $config, ?Collection $payload, int $timestamp, string $random): string
  48. {
  49. $url = get_wechat_url($config, $payload);
  50. $urlPath = parse_url($url, PHP_URL_PATH);
  51. $urlQuery = parse_url($url, PHP_URL_QUERY);
  52. return get_wechat_method($payload)."\n"
  53. .$urlPath.(empty($urlQuery) ? '' : '?'.$urlQuery)."\n"
  54. .$timestamp."\n"
  55. .$random."\n"
  56. .get_wechat_body($payload)."\n";
  57. }
  58. /**
  59. * @throws InvalidConfigException
  60. */
  61. protected function getSignature(array $config, int $timestamp, string $random, string $contents): string
  62. {
  63. $mchPublicCertPath = $config['mch_public_cert_path'] ?? null;
  64. if (empty($mchPublicCertPath)) {
  65. throw new InvalidConfigException(Exception::CONFIG_WECHAT_INVALID, '配置异常: 缺少微信配置 -- [mch_public_cert_path]');
  66. }
  67. $ssl = openssl_x509_parse(get_public_cert($mchPublicCertPath));
  68. if (empty($ssl['serialNumberHex'])) {
  69. throw new InvalidConfigException(Exception::CONFIG_WECHAT_INVALID, '配置异常: 解析微信配置 [mch_public_cert_path] 出错');
  70. }
  71. $auth = sprintf(
  72. 'mchid="%s",nonce_str="%s",timestamp="%d",serial_no="%s",signature="%s"',
  73. $config['mch_id'] ?? '',
  74. $random,
  75. $timestamp,
  76. $ssl['serialNumberHex'],
  77. get_wechat_sign($config, $contents),
  78. );
  79. return 'WECHATPAY2-SHA256-RSA2048 '.$auth;
  80. }
  81. }