1
1
import { installSnap } from '@metamask/snaps-jest' ;
2
+ import type { Hex } from '@metamask/utils' ;
2
3
import { hexToBytes } from '@noble/ciphers/utils' ;
3
4
import { secp256k1 } from '@noble/curves/secp256k1' ;
4
5
import { sha256 } from '@noble/hashes/sha256' ;
5
6
7
+ import { ERC1024 } from './utils/ERC1024' ;
8
+
6
9
describe ( 'onRpcRequest - getPublicKey' , ( ) => {
7
10
it ( 'should return this snaps public key' , async ( ) => {
8
11
const snap = await installSnap ( ) ;
@@ -19,6 +22,146 @@ describe('onRpcRequest - getPublicKey', () => {
19
22
} ) ;
20
23
} ) ;
21
24
25
+ describe ( 'onRpcRequest - getEncryptionPublicKey' , ( ) => {
26
+ it ( 'should return this snaps encryption public key' , async ( ) => {
27
+ const snap = await installSnap ( ) ;
28
+ const response = await snap . request ( {
29
+ method : 'getEncryptionPublicKey' ,
30
+ } ) ;
31
+
32
+ // E.g. length = 66 chars
33
+ // E.g. starts with '0x'
34
+ const result = 'result' in response . response && response . response . result ;
35
+ expect ( result ?. toString ( ) . length ) . toBe ( 66 ) ;
36
+ expect ( result ?. toString ( ) . startsWith ( '0x' ) ) . toBe ( true ) ;
37
+ } ) ;
38
+ } ) ;
39
+
40
+ describe ( 'onRpcRequest - decryptMessage' , ( ) => {
41
+ it ( 'should decrypt a message intended for the snaps public key' , async ( ) => {
42
+ const snap = await installSnap ( ) ;
43
+ const pkResponse = await snap . request ( {
44
+ method : 'getEncryptionPublicKey' ,
45
+ } ) ;
46
+ const publicKey = (
47
+ 'result' in pkResponse . response && pkResponse . response . result
48
+ ) ?. toString ( ) as Hex ;
49
+ const message = 'hello world' ;
50
+ const encryptedMessage = ERC1024 . encrypt ( publicKey , message ) ;
51
+ const response = await snap . request ( {
52
+ method : 'decryptMessage' ,
53
+ params : { data : encryptedMessage } ,
54
+ } ) ;
55
+
56
+ const result = 'result' in response . response && response . response . result ;
57
+ expect ( result ?. toString ( ) ) . toBe ( 'hello world' ) ;
58
+ } ) ;
59
+
60
+ it ( 'should fail to decrypt a message intended for a different recipient' , async ( ) => {
61
+ const snap = await installSnap ( ) ;
62
+ const encryptedMessage = {
63
+ version : 'x25519-xsalsa20-poly1305' ,
64
+ nonce : 'h63LvxvCOBP3x3Oou2n5JYgCM1p4p+DF' ,
65
+ ephemPublicKey : 'lmIBlLKUuSBIRjlo+/hL7ngWYpMWQ7biqk7Y6pDsaXY=' ,
66
+ ciphertext : 'g+TpY8OlU0AS9VPvaTIIqpFnWNKvWw2COSJY' ,
67
+ } ;
68
+ const response = await snap . request ( {
69
+ method : 'decryptMessage' ,
70
+ params : { data : encryptedMessage } ,
71
+ } ) ;
72
+
73
+ expect ( response ) . toRespondWithError ( {
74
+ code : - 32603 ,
75
+ message : 'invalid tag' ,
76
+ stack : expect . any ( String ) ,
77
+ } ) ;
78
+ } ) ;
79
+
80
+ it ( 'should reject a message with invalid version' , async ( ) => {
81
+ const snap = await installSnap ( ) ;
82
+ const encryptedMessage = {
83
+ version : '1' , // invalid version
84
+ nonce : 'h63LvxvCOBP3x3Oou2n5JYgCM1p4p+DF' ,
85
+ ephemPublicKey : 'lmIBlLKUuSBIRjlo+/hL7ngWYpMWQ7biqk7Y6pDsaXY=' ,
86
+ ciphertext : 'g+TpY8OlU0AS9VPvaTIIqpFnWNKvWw2COSJY' ,
87
+ } ;
88
+ const response = await snap . request ( {
89
+ method : 'decryptMessage' ,
90
+ params : { data : encryptedMessage } ,
91
+ } ) ;
92
+
93
+ expect ( response ) . toRespondWithError ( {
94
+ code : - 32602 ,
95
+ message :
96
+ '`decryptMessage`, must take a `data` parameter that must match the Eip1024EncryptedData schema' ,
97
+ stack : expect . any ( String ) ,
98
+ } ) ;
99
+ } ) ;
100
+
101
+ it ( 'should reject a message with invalid nonce' , async ( ) => {
102
+ const snap = await installSnap ( ) ;
103
+ const encryptedMessage = {
104
+ version : 'x25519-xsalsa20-poly1305' ,
105
+ nonce : 'tooshort' ,
106
+ ephemPublicKey : 'lmIBlLKUuSBIRjlo+/hL7ngWYpMWQ7biqk7Y6pDsaXY=' ,
107
+ ciphertext : 'g+TpY8OlU0AS9VPvaTIIqpFnWNKvWw2COSJY' ,
108
+ } ;
109
+ const response = await snap . request ( {
110
+ method : 'decryptMessage' ,
111
+ params : { data : encryptedMessage } ,
112
+ } ) ;
113
+
114
+ expect ( response ) . toRespondWithError ( {
115
+ code : - 32602 ,
116
+ message :
117
+ '`decryptMessage`, must take a `data` parameter that must match the Eip1024EncryptedData schema' ,
118
+ stack : expect . any ( String ) ,
119
+ } ) ;
120
+ } ) ;
121
+
122
+ it ( 'should reject a message with invalid ephemPublicKey' , async ( ) => {
123
+ const snap = await installSnap ( ) ;
124
+ const encryptedMessage = {
125
+ version : 'x25519-xsalsa20-poly1305' ,
126
+ nonce : 'h63LvxvCOBP3x3Oou2n5JYgCM1p4p+DF' ,
127
+ ephemPublicKey : 'invalid base 64' ,
128
+ ciphertext : 'g+TpY8OlU0AS9VPvaTIIqpFnWNKvWw2COSJY' ,
129
+ } ;
130
+ const response = await snap . request ( {
131
+ method : 'decryptMessage' ,
132
+ params : { data : encryptedMessage } ,
133
+ } ) ;
134
+
135
+ expect ( response ) . toRespondWithError ( {
136
+ code : - 32602 ,
137
+ message :
138
+ '`decryptMessage`, must take a `data` parameter that must match the Eip1024EncryptedData schema' ,
139
+ stack : expect . any ( String ) ,
140
+ } ) ;
141
+ } ) ;
142
+
143
+ it ( 'should reject a message with invalid params' , async ( ) => {
144
+ const snap = await installSnap ( ) ;
145
+ const encryptedMessage = JSON . stringify ( {
146
+ version : 'x25519-xsalsa20-poly1305' ,
147
+ nonce : 'h63LvxvCOBP3x3Oou2n5JYgCM1p4p+DF' ,
148
+ ephemPublicKey : 'lmIBlLKUuSBIRjlo+/hL7ngWYpMWQ7biqk7Y6pDsaXY=' ,
149
+ ciphertext : 'g+TpY8OlU0AS9VPvaTIIqpFnWNKvWw2COSJY' ,
150
+ } ) ;
151
+ const response = await snap . request ( {
152
+ method : 'decryptMessage' ,
153
+ params : { data : encryptedMessage } ,
154
+ } ) ;
155
+
156
+ expect ( response ) . toRespondWithError ( {
157
+ code : - 32602 ,
158
+ message :
159
+ '`decryptMessage`, must take a `data` parameter that must match the Eip1024EncryptedData schema' ,
160
+ stack : expect . any ( String ) ,
161
+ } ) ;
162
+ } ) ;
163
+ } ) ;
164
+
22
165
describe ( 'onRpcRequest - signMessage' , ( ) => {
23
166
it ( 'should return a signature that can be verified' , async ( ) => {
24
167
const snap = await installSnap ( ) ;
0 commit comments