4
4
5
5
namespace EasyWeChat \Tests \Pay ;
6
6
7
+ use EasyWeChat \Kernel \Support \AesEcb ;
8
+ use EasyWeChat \Kernel \Support \AesGcm ;
9
+ use EasyWeChat \Kernel \Support \Xml ;
7
10
use EasyWeChat \Pay \Contracts \Merchant ;
11
+ use EasyWeChat \Pay \Message ;
8
12
use EasyWeChat \Pay \Server ;
9
13
use EasyWeChat \Tests \TestCase ;
14
+ use Mockery \LegacyMockInterface ;
15
+ use Nyholm \Psr7 \Response ;
10
16
use Nyholm \Psr7 \ServerRequest ;
17
+ use Psr \Http \Message \ResponseInterface ;
18
+
19
+ use function bin2hex ;
20
+ use function fopen ;
21
+ use function md5 ;
22
+ use function random_bytes ;
11
23
12
24
class ServerTest extends TestCase
13
25
{
@@ -16,10 +28,13 @@ public function test_it_will_handle_validation_request()
16
28
$ request = (new ServerRequest (
17
29
'POST ' ,
18
30
'http://easywechat.com/ ' ,
19
- [],
31
+ [
32
+ 'Content-Type ' => 'application/json ' ,
33
+ ],
20
34
fopen (__DIR__ .'/../fixtures/files/pay_demo.json ' , 'r ' )
21
35
));
22
36
37
+ /** @var Merchant&LegacyMockInterface $merchant */
23
38
$ merchant = \Mockery::mock (Merchant::class);
24
39
$ merchant ->shouldReceive ('getSecretKey ' )->andReturn ('key ' );
25
40
@@ -28,4 +43,106 @@ public function test_it_will_handle_validation_request()
28
43
$ response = $ server ->serve ();
29
44
$ this ->assertSame ('{"code":"SUCCESS","message":"成功"} ' , \strval ($ response ->getBody ()));
30
45
}
46
+
47
+ public function test_legacy_encryped_by_aesecb_refund_request ()
48
+ {
49
+ /** @var Merchant&LegacyMockInterface $merchant */
50
+ $ merchant = \Mockery::mock (Merchant::class);
51
+ $ merchant ->shouldReceive (['getV2SecretKey ' => random_bytes (32 )]);
52
+ $ symmtricKey = $ merchant ->getV2SecretKey ();
53
+
54
+ $ server = new Server ($ merchant , new ServerRequest (
55
+ 'POST ' ,
56
+ 'http://easywechat.com/sample-webhook-handler ' ,
57
+ [
58
+ 'Content-Type ' => 'text/xml ' ,
59
+ ],
60
+ Xml::build ([
61
+ 'return_code ' => 'SUCCESS ' ,
62
+ 'req_info ' => AesEcb::encrypt (Xml::build ([
63
+ 'refund_id ' => '50000408942018111907145868882 ' ,
64
+ 'transaction_id ' => '4200000215201811190261405420 ' ,
65
+ ]), md5 ($ symmtricKey ), '' ),
66
+ ])
67
+ ));
68
+
69
+ $ response = $ server ->with (function (Message $ message ): ResponseInterface {
70
+ $ source = $ message ->getOriginalContents ();
71
+ $ parsed = $ message ->toArray ();
72
+
73
+ $ this ->assertStringContainsString ('<xml> ' , $ source );
74
+ $ this ->assertStringContainsString ('<req_info> ' , $ source );
75
+ $ this ->assertStringNotContainsString ('<refund_id> ' , $ source );
76
+ $ this ->assertStringNotContainsString ('<transaction_id> ' , $ source );
77
+ $ this ->assertArrayNotHasKey ('return_code ' , $ parsed );
78
+ $ this ->assertArrayNotHasKey ('req_info ' , $ parsed );
79
+ $ this ->assertArrayHasKey ('refund_id ' , $ parsed );
80
+ $ this ->assertArrayHasKey ('transaction_id ' , $ parsed );
81
+
82
+ return new Response (
83
+ 200 ,
84
+ ['Content-Type ' => 'text/xml ' ],
85
+ '<xml><return_code>SUCCESS</return_code></xml> '
86
+ );
87
+ })->serve ();
88
+
89
+ $ this ->assertEquals (200 , $ response ->getStatusCode ());
90
+ $ this ->assertSame ('<xml><return_code>SUCCESS</return_code></xml> ' , \strval ($ response ->getBody ()));
91
+ }
92
+
93
+ public function test_legacy_encryped_by_aesgcm_notification_request ()
94
+ {
95
+ /** @var Merchant&LegacyMockInterface $merchant */
96
+ $ merchant = \Mockery::mock (Merchant::class);
97
+ $ merchant ->shouldReceive (['getSecretKey ' => random_bytes (32 )]);
98
+ $ symmtricKey = $ merchant ->getSecretKey ();
99
+
100
+ $ server = new Server ($ merchant , new ServerRequest (
101
+ 'POST ' ,
102
+ 'http://easywechat.com/sample-webhook-handler ' ,
103
+ [
104
+ 'Content-Type ' => 'text/xml ' ,
105
+ ],
106
+ Xml::build ([
107
+ 'event_type ' => 'TRANSACTION.SUCCESS ' ,
108
+ 'event_algorithm ' => 'AEAD_AES_256_GCM ' ,
109
+ 'event_nonce ' => $ nonce = bin2hex (random_bytes (6 )),
110
+ 'event_associated_data ' => $ aad = '' ,
111
+ 'event_ciphertext ' => AesGcm::encrypt (Xml::build ([
112
+ 'state ' => 'USER_PAID ' ,
113
+ 'service_id ' => '1234352342 ' ,
114
+ ]), $ symmtricKey , iv: $ nonce , aad: $ aad ),
115
+ ])
116
+ ));
117
+
118
+ $ response = $ server ->with (function (Message $ message ): ResponseInterface {
119
+ $ source = $ message ->getOriginalContents ();
120
+ $ parsed = $ message ->toArray ();
121
+
122
+ $ this ->assertStringContainsString ('<xml> ' , $ source );
123
+ $ this ->assertStringContainsString ('<event_type> ' , $ source );
124
+ $ this ->assertStringContainsString ('<event_algorithm> ' , $ source );
125
+ $ this ->assertStringContainsString ('<event_nonce> ' , $ source );
126
+ $ this ->assertStringContainsString ('<event_associated_data> ' , $ source );
127
+ $ this ->assertStringContainsString ('<event_ciphertext> ' , $ source );
128
+ $ this ->assertStringNotContainsString ('<state> ' , $ source );
129
+ $ this ->assertStringNotContainsString ('<service_id> ' , $ source );
130
+ $ this ->assertArrayHasKey ('event_type ' , $ parsed );
131
+ $ this ->assertArrayHasKey ('event_algorithm ' , $ parsed );
132
+ $ this ->assertArrayHasKey ('event_nonce ' , $ parsed );
133
+ $ this ->assertArrayHasKey ('event_associated_data ' , $ parsed );
134
+ $ this ->assertArrayHasKey ('event_ciphertext ' , $ parsed );
135
+ $ this ->assertArrayHasKey ('state ' , $ parsed );
136
+ $ this ->assertArrayHasKey ('service_id ' , $ parsed );
137
+
138
+ return new Response (
139
+ 500 ,
140
+ ['Content-Type ' => 'text/xml ' ],
141
+ '<xml><code>ERROR_NAME</code><message>ERROR_DESCRIPTION</message></xml> '
142
+ );
143
+ })->serve ();
144
+
145
+ $ this ->assertEquals (500 , $ response ->getStatusCode ());
146
+ $ this ->assertSame ('<xml><code>ERROR_NAME</code><message>ERROR_DESCRIPTION</message></xml> ' , \strval ($ response ->getBody ()));
147
+ }
31
148
}
0 commit comments