Passed
Pull Request — master (#1002)
by
unknown
02:06
created

CallbackPlugin   A

Complexity

Total Complexity 7

Size/Duplication

Total Lines 59
Duplicated Lines 0 %

Importance

Changes 1
Bugs 0 Features 0
Metric Value
eloc 29
c 1
b 0
f 0
dl 0
loc 59
rs 10
wmc 7

3 Methods

Rating   Name   Duplication   Size   Complexity  
A assembly() 0 23 1
A verifySign() 0 19 4
A formatRequestAndParams() 0 7 2
1
<?php
2
3
declare(strict_types=1);
4
5
namespace Yansongda\Pay\Plugin\Epay;
6
7
use Closure;
8
use Yansongda\Artful\Contract\PluginInterface;
9
use Yansongda\Artful\Direction\NoHttpRequestDirection;
10
use Yansongda\Artful\Exception\InvalidConfigException;
11
use Yansongda\Artful\Exception\InvalidParamsException;
12
use Yansongda\Artful\Exception\InvalidResponseException;
13
use Yansongda\Artful\Logger;
14
use Yansongda\Artful\Rocket;
15
use Yansongda\Pay\Exception\Exception;
16
use Yansongda\Supports\Collection;
17
18
use function Yansongda\Pay\get_provider_config;
19
20
class CallbackPlugin implements PluginInterface
21
{
22
    public function assembly(Rocket $rocket, Closure $next): Rocket
23
    {
24
        Logger::info('[epay][CallbackPlugin] 插件开始装载', ['rocket' => $rocket]);
25
26
        $this->formatRequestAndParams($rocket);
27
28
        $params = $rocket->getParams();
29
        $config = get_provider_config('epay', $params);
30
31
        $payload = $rocket->getPayload();
32
        $signature = $payload->get('sign');
33
34
        $payload->forget('sign');
35
        $payload->forget('signType');
36
37
        $this->verifySign($config, $payload, $signature);
38
39
        $rocket->setDirection(NoHttpRequestDirection::class)
40
            ->setDestination($rocket->getPayload());
41
42
        Logger::info('[epay][CallbackPlugin] 插件装载完毕', ['rocket' => $rocket]);
43
44
        return $next($rocket);
45
    }
46
47
    protected function verifySign(array $config, Collection $payload, ?string $signature = null): void
48
    {
49
        if (!$signature) {
50
            throw new InvalidResponseException(Exception::SIGN_ERROR, 'Verify Epay payload Sign Failed: sign is empty', $payload);
51
        }
52
53
        $publicCert = $config['epay_public_cert_path'] ?? null;
54
55
        if (empty($publicCert)) {
56
            throw new InvalidConfigException(Exception::CONFIG_EPAY_INVALID, 'Missing Epay Config -- [epay_public_cert_path]');
57
        }
58
59
        $result = 1 === openssl_verify(
60
            $payload->sortKeys()->toString(),
61
            base64_decode($signature),
62
            file_get_contents($publicCert)
63
        );
64
        if (!$result) {
65
            throw new InvalidConfigException(Exception::SIGN_ERROR, 'Verify Epay Response Sign Failed', func_get_args());
66
        }
67
    }
68
69
    /**
70
     * @throws InvalidParamsException
71
     */
72
    protected function formatRequestAndParams(Rocket $rocket): void
73
    {
74
        $request = $rocket->getParams()['request'] ?? null;
75
        if (!$request instanceof Collection) {
76
            throw new InvalidParamsException(Exception::PARAMS_CALLBACK_REQUEST_INVALID);
77
        }
78
        $rocket->setPayload($request)->setParams($rocket->getParams()['params'] ?? []);
79
    }
80
}
81