@notabene/verify-proof 1.11.0 → 1.11.1-next.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (75) hide show
  1. package/README.md +21 -15
  2. package/dist/bitcoin-3CW4MNAW.cjs +314 -0
  3. package/dist/bitcoin-3CW4MNAW.cjs.map +1 -0
  4. package/dist/bitcoin-QK53ILBF.js +312 -0
  5. package/dist/bitcoin-QK53ILBF.js.map +1 -0
  6. package/dist/cardano-DYBYEAAF.cjs +53 -0
  7. package/dist/cardano-DYBYEAAF.cjs.map +1 -0
  8. package/dist/cardano-WE6YXYLW.js +31 -0
  9. package/dist/cardano-WE6YXYLW.js.map +1 -0
  10. package/dist/chunk-E3V5ATTC.js +38 -0
  11. package/dist/chunk-E3V5ATTC.js.map +1 -0
  12. package/dist/chunk-OAXNH5XR.cjs +42 -0
  13. package/dist/chunk-OAXNH5XR.cjs.map +1 -0
  14. package/dist/concordium-HQC37GCK.cjs +188 -0
  15. package/dist/concordium-HQC37GCK.cjs.map +1 -0
  16. package/dist/concordium-XX4XYLLU.js +186 -0
  17. package/dist/concordium-XX4XYLLU.js.map +1 -0
  18. package/dist/cosmos-64MKE5FJ.cjs +683 -0
  19. package/dist/cosmos-64MKE5FJ.cjs.map +1 -0
  20. package/dist/cosmos-QMH7BK7S.js +681 -0
  21. package/dist/cosmos-QMH7BK7S.js.map +1 -0
  22. package/dist/eth-3DX3PXDU.cjs +37 -0
  23. package/dist/eth-3DX3PXDU.cjs.map +1 -0
  24. package/dist/eth-ICLGRJE5.js +34 -0
  25. package/dist/eth-ICLGRJE5.js.map +1 -0
  26. package/dist/index.cjs +69 -1
  27. package/dist/index.cjs.map +1 -1
  28. package/dist/index.d.cts +5 -0
  29. package/dist/index.d.ts +5 -2
  30. package/dist/index.js +67 -1
  31. package/dist/index.js.map +1 -1
  32. package/dist/solana-4KQFLZUC.js +342 -0
  33. package/dist/solana-4KQFLZUC.js.map +1 -0
  34. package/dist/solana-PQ5K4NGO.cjs +365 -0
  35. package/dist/solana-PQ5K4NGO.cjs.map +1 -0
  36. package/dist/tron-F5AARBY4.cjs +58 -0
  37. package/dist/tron-F5AARBY4.cjs.map +1 -0
  38. package/dist/tron-OBLPB2LN.js +53 -0
  39. package/dist/tron-OBLPB2LN.js.map +1 -0
  40. package/dist/xlm-5GODWWL2.cjs +28 -0
  41. package/dist/xlm-5GODWWL2.cjs.map +1 -0
  42. package/dist/xlm-GX2QGFLI.js +26 -0
  43. package/dist/xlm-GX2QGFLI.js.map +1 -0
  44. package/dist/xrpl-7HQLIDAK.cjs +1174 -0
  45. package/dist/xrpl-7HQLIDAK.cjs.map +1 -0
  46. package/dist/xrpl-YCDFXBGQ.js +1169 -0
  47. package/dist/xrpl-YCDFXBGQ.js.map +1 -0
  48. package/package.json +11 -2
  49. package/src/cardano.ts +2 -2
  50. package/src/index.ts +31 -19
  51. package/src/solana.ts +1 -1
  52. package/src/tests/solana.test.ts +1 -1
  53. package/dist/bitcoin.d.ts +0 -2
  54. package/dist/cardano.d.ts +0 -2
  55. package/dist/concordium.d.ts +0 -15
  56. package/dist/cosmos.d.ts +0 -2
  57. package/dist/eth.d.ts +0 -4
  58. package/dist/index.modern.js +0 -2
  59. package/dist/index.modern.js.map +0 -1
  60. package/dist/index.umd.js +0 -2
  61. package/dist/index.umd.js.map +0 -1
  62. package/dist/solana.d.ts +0 -17
  63. package/dist/tests/bitcoin.test.d.ts +0 -1
  64. package/dist/tests/cardano.test.d.ts +0 -1
  65. package/dist/tests/concordium.test.d.ts +0 -1
  66. package/dist/tests/cosmos.test.d.ts +0 -1
  67. package/dist/tests/eth.test.d.ts +0 -1
  68. package/dist/tests/index.test.d.ts +0 -1
  69. package/dist/tests/solana.test.d.ts +0 -1
  70. package/dist/tests/tron.test.d.ts +0 -1
  71. package/dist/tests/xlm.test.d.ts +0 -1
  72. package/dist/tests/xrpl.test.d.ts +0 -1
  73. package/dist/tron.d.ts +0 -6
  74. package/dist/xlm.d.ts +0 -2
  75. package/dist/xrpl.d.ts +0 -5
package/README.md CHANGED
@@ -32,16 +32,28 @@ async function checkProof(proof: OwnershipProof) {
32
32
  }
33
33
  ```
34
34
 
35
+ ### SSR/ESM Compatibility
36
+
37
+ This library is fully compatible with modern SSR frameworks like SvelteKit, Next.js, and others. It uses dynamic imports to ensure that cryptographic dependencies (especially Bitcoin libraries) are loaded on-demand, preventing `Buffer is not defined` errors in SSR environments.
38
+
39
+ The library automatically sets up a `Buffer` polyfill for browser/SSR environments before any crypto code executes. No additional configuration is required in your consuming application.
40
+
35
41
  ## Supported Proof Types
36
42
 
37
- - Checkbox Confirmation (`ProofTypes.CheckboxConfirmation`)
43
+ - Self Declaration (`ProofTypes.SelfDeclaration`)
38
44
  - Screenshot (`ProofTypes.Screenshot`)
39
- - Personal Sign EIP191 (`ProofTypes.EIP191`)
40
- - Personal Sign EIP712 (`ProofTypes.EIP712`)
41
- - Personal Sign BIP137 (`ProofTypes.BIP137`)
42
- - BIP322 Signature (`ProofTypes.BIP322`)
43
- - Personal Sign XPUB (`ProofTypes.XPUB`)
44
- - Micro Transfer (`ProofTypes.MicroTransfer`)
45
+ - Ethereum EIP191 (`ProofTypes.EIP191`)
46
+ - Ethereum EIP712 (`ProofTypes.EIP712`)
47
+ - Bitcoin BIP137 (`ProofTypes.BIP137`)
48
+ - Bitcoin BIP322 (`ProofTypes.BIP322`)
49
+ - Solana ED25519 (`ProofTypes.ED25519`)
50
+ - Solana SIWS (`ProofTypes.SOL_SIWX`)
51
+ - Cardano CIP8 (`ProofTypes.CIP8`)
52
+ - XRPL ED25519 (`ProofTypes.XRP_ED25519`)
53
+ - Stellar XLM ED25519 (`ProofTypes.XLM_ED25519`)
54
+ - Concordium (`ProofTypes.CONCORDIUM`)
55
+ - Cosmos (`ProofTypes.COSMOS`)
56
+ - Tron TIP191 (`ProofTypes.TIP191`)
45
57
 
46
58
  ## Development
47
59
 
@@ -53,16 +65,10 @@ To set up the development environment:
53
65
 
54
66
  ## Scripts
55
67
 
56
- - `yarn dev`: Start development server
57
68
  - `yarn build`: Build the project
58
69
  - `yarn lint`: Run linter
59
- - `yarn test`: Run tests
60
-
61
- ## Dependencies
62
-
63
- - `@notabene/javascript-sdk`: Notabene JavaScript SDK
64
- - `viem`: Ethereum utility library
65
- - `bitcoinjs-message`: Bitcoin message signing and verification
70
+ - `yarn test:unit`: Run tests
71
+ - `yarn test:coverage`: Run tests with coverage
66
72
 
67
73
  ## License
68
74
 
@@ -0,0 +1,314 @@
1
+ 'use strict';
2
+
3
+ var buffer = require('buffer');
4
+ require('./chunk-OAXNH5XR.cjs');
5
+ var javascriptSdk = require('@notabene/javascript-sdk');
6
+ var varuintBitcoin = require('varuint-bitcoin');
7
+ var base = require('@scure/base');
8
+ var ox = require('ox');
9
+ var secp256k1 = require('@noble/curves/secp256k1');
10
+ var bip322Js = require('bip322-js');
11
+
12
+ if (typeof globalThis !== 'undefined' && !globalThis.Buffer) { globalThis.Buffer = buffer.Buffer; }
13
+ var CHAIN_CONFIGS = {
14
+ bitcoin: {
15
+ messagePrefix: "Bitcoin Signed Message:\n",
16
+ pubKeyHashVersion: 0,
17
+ // 1...
18
+ scriptHashVersion: 5,
19
+ // 3...
20
+ bech32Prefix: "bc",
21
+ isTestnet: false
22
+ },
23
+ bitcoincash: {
24
+ messagePrefix: "Bitcoin Signed Message:\n",
25
+ pubKeyHashVersion: 0,
26
+ // 1...
27
+ scriptHashVersion: 5,
28
+ // 3...
29
+ bech32Prefix: "bc",
30
+ isTestnet: false
31
+ },
32
+ litecoin: {
33
+ messagePrefix: "Litecoin Signed Message:\n",
34
+ pubKeyHashVersion: 48,
35
+ // L... or M...
36
+ scriptHashVersion: 50,
37
+ // 3... or M...
38
+ bech32Prefix: "ltc",
39
+ isTestnet: false
40
+ },
41
+ dogecoin: {
42
+ messagePrefix: "Dogecoin Signed Message:\n",
43
+ pubKeyHashVersion: 30,
44
+ // D...
45
+ scriptHashVersion: 22,
46
+ // A...
47
+ isTestnet: false
48
+ },
49
+ dash: {
50
+ messagePrefix: "DarkCoin Signed Message:\n",
51
+ pubKeyHashVersion: 76,
52
+ // X...
53
+ scriptHashVersion: 16,
54
+ // 7...
55
+ isTestnet: false
56
+ },
57
+ zcash: {
58
+ messagePrefix: "Zcash Signed Message:\n",
59
+ pubKeyHashVersion: Uint8Array.from([28, 184]),
60
+ // <-- FIXED
61
+ scriptHashVersion: Uint8Array.from([28, 189]),
62
+ isTestnet: false
63
+ },
64
+ testnet: {
65
+ messagePrefix: "Bitcoin Signed Message:\n",
66
+ pubKeyHashVersion: 111,
67
+ // m or n
68
+ scriptHashVersion: 196,
69
+ // 2
70
+ bech32Prefix: "tb",
71
+ isTestnet: true
72
+ }
73
+ };
74
+ async function verifyBTCSignature(proof) {
75
+ const [ns, , address] = proof.address.split(/:/);
76
+ if (ns !== "bip122") return { ...proof, status: javascriptSdk.ProofStatus.FAILED };
77
+ const chainConfig = getChainConfig(address);
78
+ if (!chainConfig) return { ...proof, status: javascriptSdk.ProofStatus.FAILED };
79
+ const isZcash = address.startsWith("t1") || address.startsWith("t3");
80
+ if (isZcash) {
81
+ return verifyBIP137(address, proof, chainConfig);
82
+ }
83
+ if (chainConfig.isTestnet) {
84
+ return verifyBIP322(address, proof);
85
+ }
86
+ const isTaproot = address.startsWith("bc1p") || address.startsWith("tb1p");
87
+ if (isTaproot && proof.type === javascriptSdk.ProofTypes.BIP137) {
88
+ return verifyBIP322(address, proof);
89
+ }
90
+ try {
91
+ switch (proof.type) {
92
+ case javascriptSdk.ProofTypes.BIP137:
93
+ return verifyBIP137(address, proof, chainConfig);
94
+ case javascriptSdk.ProofTypes.BIP322:
95
+ return verifyBIP322(address, proof);
96
+ default:
97
+ return {
98
+ ...proof,
99
+ status: javascriptSdk.ProofStatus.FAILED
100
+ };
101
+ }
102
+ } catch {
103
+ return {
104
+ ...proof,
105
+ status: javascriptSdk.ProofStatus.FAILED
106
+ };
107
+ }
108
+ }
109
+ function getChainConfig(address) {
110
+ if (address.startsWith("1") || address.startsWith("3") || address.startsWith("bc1")) {
111
+ return CHAIN_CONFIGS["bitcoin"];
112
+ }
113
+ if (address.startsWith("t1") || address.startsWith("t3")) {
114
+ return CHAIN_CONFIGS["zcash"];
115
+ }
116
+ if (address.startsWith("L") || address.startsWith("M") || address.startsWith("ltc1")) {
117
+ return CHAIN_CONFIGS["litecoin"];
118
+ }
119
+ if (address.startsWith("D") || address.startsWith("A")) {
120
+ return CHAIN_CONFIGS["dogecoin"];
121
+ }
122
+ if (address.startsWith("X") || address.startsWith("7")) {
123
+ return CHAIN_CONFIGS["dash"];
124
+ }
125
+ if (address.startsWith("q")) {
126
+ return CHAIN_CONFIGS["bitcoincash"];
127
+ }
128
+ if (address.startsWith("tb1")) {
129
+ return CHAIN_CONFIGS["testnet"];
130
+ }
131
+ return CHAIN_CONFIGS["bitcoin"];
132
+ }
133
+ function verifyBIP322(address, proof) {
134
+ const { attestation, proof: signatureProof } = proof;
135
+ const verified = bip322Js.Verifier.verifySignature(
136
+ address,
137
+ attestation,
138
+ signatureProof
139
+ );
140
+ return {
141
+ ...proof,
142
+ status: verified ? javascriptSdk.ProofStatus.VERIFIED : javascriptSdk.ProofStatus.FAILED
143
+ };
144
+ }
145
+ function verifyBIP137(address, proof, chainConfig) {
146
+ const derivationMode = getDerivationMode(address);
147
+ const useSegwitEncoding = Boolean(
148
+ chainConfig.bech32Prefix && (derivationMode === "Native SegWit" /* NATIVE */ || derivationMode === "SegWit" /* SEGWIT */ && !address.startsWith("1"))
149
+ );
150
+ const verified = verify(
151
+ proof.attestation,
152
+ address,
153
+ proof.proof,
154
+ useSegwitEncoding,
155
+ chainConfig
156
+ );
157
+ return {
158
+ ...proof,
159
+ status: verified ? javascriptSdk.ProofStatus.VERIFIED : javascriptSdk.ProofStatus.FAILED
160
+ };
161
+ }
162
+ function getDerivationMode(address) {
163
+ if (address.match("^(bc1|tb1|ltc1).*")) {
164
+ return "Native SegWit" /* NATIVE */;
165
+ } else if (address.match("^[32M].*")) {
166
+ return "SegWit" /* SEGWIT */;
167
+ } else if (address.match("^[1nmL].*")) {
168
+ return "Legacy" /* LEGACY */;
169
+ } else if (address.match("^(D).*")) {
170
+ return "Dogecoin" /* DOGECOIN */;
171
+ } else if (address.match("^(q).*")) {
172
+ return "Bitcoin Cash" /* BCH */;
173
+ } else if (address.match("^(t1|t3).*")) {
174
+ return "Legacy" /* LEGACY */;
175
+ } else if (address.match("^[X7].*")) {
176
+ return "Legacy" /* LEGACY */;
177
+ } else {
178
+ throw new Error(
179
+ "INVALID ADDRESS: ".concat(address).concat(" is not a valid or a supported address")
180
+ );
181
+ }
182
+ }
183
+ function decodeSignature(proof) {
184
+ const sigbytes = base.base64.decode(proof);
185
+ if (sigbytes.length !== 65) throw new Error("Invalid signature length");
186
+ const flagByte = sigbytes[0] - 27;
187
+ if (flagByte > 15 || flagByte < 0) {
188
+ throw new Error("Invalid signature parameter");
189
+ }
190
+ const compressed = !!(flagByte & 12);
191
+ const recovery = flagByte & 3;
192
+ const signature = secp256k1.secp256k1.Signature.fromCompact(sigbytes.slice(1));
193
+ return {
194
+ compressed,
195
+ segwitType: !(flagByte & 8) ? void 0 : !(flagByte & 4) ? "p2sh(p2wpkh)" /* P2SH_P2WPKH */ : "p2wpkh" /* P2WPKH */,
196
+ signature: signature.addRecoveryBit(recovery)
197
+ };
198
+ }
199
+ function verify(attestation, address, proof, checkSegwitAlways, chainConfig) {
200
+ const { compressed, segwitType, signature } = decodeSignature(proof);
201
+ if (checkSegwitAlways && !compressed) {
202
+ throw new Error(
203
+ "checkSegwitAlways can only be used with a compressed pubkey signature flagbyte"
204
+ );
205
+ }
206
+ const hash = magicHash(attestation, chainConfig.messagePrefix);
207
+ const publicKey = signature.recoverPublicKey(hash);
208
+ const publicKeyBytes = publicKey.toRawBytes(compressed);
209
+ const publicKeyHash = hash160(publicKeyBytes);
210
+ let actual = "";
211
+ if (address.startsWith("q")) {
212
+ actual = encodeBase58AddressFormat(
213
+ chainConfig.pubKeyHashVersion,
214
+ publicKeyHash
215
+ );
216
+ return actual.startsWith("1");
217
+ }
218
+ if (segwitType) {
219
+ if (segwitType === "p2sh(p2wpkh)" /* P2SH_P2WPKH */) {
220
+ actual = encodeBase58AddressFormat(
221
+ chainConfig.scriptHashVersion,
222
+ publicKeyHash
223
+ );
224
+ } else {
225
+ if (chainConfig.bech32Prefix) {
226
+ actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);
227
+ } else {
228
+ actual = encodeBase58AddressFormat(
229
+ chainConfig.scriptHashVersion,
230
+ publicKeyHash
231
+ );
232
+ }
233
+ }
234
+ } else {
235
+ if (address.startsWith("3") && !segwitType) {
236
+ const redeemScript = new Uint8Array(22);
237
+ redeemScript[0] = 0;
238
+ redeemScript[1] = 20;
239
+ redeemScript.set(publicKeyHash, 2);
240
+ const redeemScriptHash = hash160(redeemScript);
241
+ const p2shP2wpkh = encodeBase58AddressFormat(
242
+ chainConfig.scriptHashVersion,
243
+ redeemScriptHash
244
+ );
245
+ const legacyP2sh = encodeBase58AddressFormat(
246
+ chainConfig.scriptHashVersion,
247
+ publicKeyHash
248
+ );
249
+ if (address === p2shP2wpkh || address === legacyP2sh) {
250
+ return true;
251
+ }
252
+ actual = legacyP2sh;
253
+ } else if (address.startsWith("bc1q") || address.startsWith("tb1q") || address.startsWith("ltc1q")) {
254
+ if (chainConfig.bech32Prefix) {
255
+ actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);
256
+ } else {
257
+ actual = encodeBase58AddressFormat(
258
+ chainConfig.pubKeyHashVersion,
259
+ publicKeyHash
260
+ );
261
+ }
262
+ } else if (checkSegwitAlways && chainConfig.bech32Prefix) {
263
+ try {
264
+ actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);
265
+ } catch (e) {
266
+ actual = encodeBase58AddressFormat(
267
+ chainConfig.scriptHashVersion,
268
+ publicKeyHash
269
+ );
270
+ }
271
+ } else {
272
+ actual = encodeBase58AddressFormat(
273
+ chainConfig.pubKeyHashVersion,
274
+ publicKeyHash
275
+ );
276
+ }
277
+ }
278
+ return actual === address;
279
+ }
280
+ var base58check = base.createBase58check(ox.Hash.sha256);
281
+ function encodeBase58AddressFormat(version, publicKeyHash) {
282
+ const prefixBytes = typeof version === "number" ? Uint8Array.of(version) : version;
283
+ const payload = new Uint8Array(prefixBytes.length + publicKeyHash.length);
284
+ payload.set(prefixBytes);
285
+ payload.set(publicKeyHash, prefixBytes.length);
286
+ return base58check.encode(payload);
287
+ }
288
+ function magicHash(attestation, messagePrefix) {
289
+ const prefix = new TextEncoder().encode(messagePrefix);
290
+ const message = new TextEncoder().encode(attestation);
291
+ const length = varuintBitcoin.encode(message.length).buffer;
292
+ const buffer = new Uint8Array(
293
+ prefix.length + length.byteLength + message.length
294
+ );
295
+ buffer.set(prefix);
296
+ buffer.set(new Uint8Array(length), prefix.length);
297
+ buffer.set(message, prefix.length + length.byteLength);
298
+ return hash256(buffer);
299
+ }
300
+ function encodeBech32Address(publicKeyHash, prefix = "bc") {
301
+ const bwords = base.bech32.toWords(publicKeyHash);
302
+ bwords.unshift(0);
303
+ return base.bech32.encode(prefix, bwords);
304
+ }
305
+ function hash256(buffer) {
306
+ return ox.Hash.sha256(ox.Hash.sha256(buffer));
307
+ }
308
+ function hash160(buffer) {
309
+ return ox.Hash.ripemd160(ox.Hash.sha256(buffer));
310
+ }
311
+
312
+ exports.verifyBTCSignature = verifyBTCSignature;
313
+ //# sourceMappingURL=bitcoin-3CW4MNAW.cjs.map
314
+ //# sourceMappingURL=bitcoin-3CW4MNAW.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/bitcoin.ts"],"names":["ProofStatus","ProofTypes","Verifier","base64","secp256k1","createBase58check","Hash","encodeLength","bech32"],"mappings":"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","file":"bitcoin-3CW4MNAW.cjs","sourcesContent":["import {\n ProofStatus,\n ProofTypes,\n SignatureProof,\n} from \"@notabene/javascript-sdk\";\n\nimport { encode as encodeLength } from \"varuint-bitcoin\";\nimport { base64, bech32, createBase58check } from \"@scure/base\";\nimport { Hash } from \"ox\";\nimport { secp256k1 } from \"@noble/curves/secp256k1\";\nimport { SignatureType } from \"@noble/curves/abstract/weierstrass\";\nimport { Verifier } from \"bip322-js\";\n\nenum SEGWIT_TYPES {\n P2WPKH = \"p2wpkh\",\n P2SH_P2WPKH = \"p2sh(p2wpkh)\",\n}\n\ninterface ChainConfig {\n messagePrefix: string;\n pubKeyHashVersion: number | Uint8Array;\n scriptHashVersion: number | Uint8Array;\n bech32Prefix?: string;\n isTestnet?: boolean;\n}\n\nconst CHAIN_CONFIGS: Record<string, ChainConfig> = {\n bitcoin: {\n messagePrefix: \"\\u0018Bitcoin Signed Message:\\n\",\n pubKeyHashVersion: 0x00, // 1...\n scriptHashVersion: 0x05, // 3...\n bech32Prefix: \"bc\",\n isTestnet: false,\n },\n bitcoincash: {\n messagePrefix: \"\\u0018Bitcoin Signed Message:\\n\",\n pubKeyHashVersion: 0x00, // 1...\n scriptHashVersion: 0x05, // 3...\n bech32Prefix: \"bc\",\n isTestnet: false,\n },\n litecoin: {\n messagePrefix: \"\\u0019Litecoin Signed Message:\\n\",\n pubKeyHashVersion: 0x30, // L... or M...\n scriptHashVersion: 0x32, // 3... or M...\n bech32Prefix: \"ltc\",\n isTestnet: false,\n },\n dogecoin: {\n messagePrefix: \"\\u0019Dogecoin Signed Message:\\n\",\n pubKeyHashVersion: 0x1e, // D...\n scriptHashVersion: 0x16, // A...\n isTestnet: false,\n },\n dash: {\n messagePrefix: \"\\u0019DarkCoin Signed Message:\\n\",\n pubKeyHashVersion: 0x4c, // X...\n scriptHashVersion: 0x10, // 7...\n isTestnet: false,\n },\n zcash: {\n messagePrefix: \"\\u0018Zcash Signed Message:\\n\",\n pubKeyHashVersion: Uint8Array.from([0x1c, 0xb8]), // <-- FIXED\n scriptHashVersion: Uint8Array.from([0x1c, 0xbd]),\n isTestnet: false,\n },\n\n testnet: {\n messagePrefix: \"\\u0018Bitcoin Signed Message:\\n\",\n pubKeyHashVersion: 0x6f, // m or n\n scriptHashVersion: 0xc4, // 2\n bech32Prefix: \"tb\",\n isTestnet: true,\n },\n};\n\nenum DerivationMode {\n LEGACY = \"Legacy\",\n NATIVE = \"Native SegWit\",\n SEGWIT = \"SegWit\",\n P2SH_SEGWIT = \"p2sh\",\n BCH = \"Bitcoin Cash\",\n ETHEREUM = \"Ethereum\",\n DOGECOIN = \"Dogecoin\",\n UNKNOWN = \"Unknown\",\n}\n\nexport async function verifyBTCSignature(\n proof: SignatureProof\n): Promise<SignatureProof> {\n const [ns, , address] = proof.address.split(/:/);\n if (ns !== \"bip122\") return { ...proof, status: ProofStatus.FAILED };\n\n // Map chainId to our chain configuration\n const chainConfig = getChainConfig(address);\n if (!chainConfig) return { ...proof, status: ProofStatus.FAILED };\n\n const isZcash = address.startsWith(\"t1\") || address.startsWith(\"t3\");\n if (isZcash) {\n return verifyBIP137(address, proof, chainConfig);\n }\n\n // Use BIP322 for testnet addresses\n if (chainConfig.isTestnet) {\n return verifyBIP322(address, proof);\n }\n\n // Check if this is a Taproot address (bc1p or tb1p)\n const isTaproot = address.startsWith(\"bc1p\") || address.startsWith(\"tb1p\");\n\n // For Taproot addresses with BIP-137 proof type, use BIP-322 verification\n // since BIP-137 doesn't officially support Taproot\n if (isTaproot && proof.type === ProofTypes.BIP137) {\n return verifyBIP322(address, proof);\n }\n\n try {\n switch (proof.type) {\n case ProofTypes.BIP137:\n return verifyBIP137(address, proof, chainConfig);\n case ProofTypes.BIP322:\n return verifyBIP322(address, proof);\n default:\n return {\n ...proof,\n status: ProofStatus.FAILED,\n };\n }\n } catch {\n return {\n ...proof,\n status: ProofStatus.FAILED,\n };\n }\n}\n\nfunction getChainConfig(address: string): ChainConfig {\n if (\n address.startsWith(\"1\") ||\n address.startsWith(\"3\") ||\n address.startsWith(\"bc1\")\n ) {\n return CHAIN_CONFIGS[\"bitcoin\"];\n }\n if (address.startsWith(\"t1\") || address.startsWith(\"t3\")) {\n return CHAIN_CONFIGS[\"zcash\"];\n }\n if (\n address.startsWith(\"L\") ||\n address.startsWith(\"M\") ||\n address.startsWith(\"ltc1\")\n ) {\n return CHAIN_CONFIGS[\"litecoin\"];\n }\n if (address.startsWith(\"D\") || address.startsWith(\"A\")) {\n return CHAIN_CONFIGS[\"dogecoin\"];\n }\n if (address.startsWith(\"X\") || address.startsWith(\"7\")) {\n return CHAIN_CONFIGS[\"dash\"];\n }\n if (address.startsWith(\"q\")) {\n return CHAIN_CONFIGS[\"bitcoincash\"];\n }\n if (address.startsWith(\"tb1\")) {\n return CHAIN_CONFIGS[\"testnet\"];\n }\n\n return CHAIN_CONFIGS[\"bitcoin\"];\n}\n\nfunction verifyBIP322(address: string, proof: SignatureProof) {\n const { attestation, proof: signatureProof } = proof;\n const verified = Verifier.verifySignature(\n address,\n attestation,\n signatureProof\n );\n return {\n ...proof,\n status: verified ? ProofStatus.VERIFIED : ProofStatus.FAILED,\n };\n}\n\nfunction verifyBIP137(\n address: string,\n proof: SignatureProof,\n chainConfig: ChainConfig\n) {\n const derivationMode = getDerivationMode(address);\n\n // For legacy addresses (starting with \"1\"), never use SegWit encoding\n // For P2SH addresses (starting with \"3\"), use SegWit encoding if they have bech32 support\n // For native SegWit addresses (bc1, tb1, ltc1), always use SegWit encoding\n const useSegwitEncoding = Boolean(\n chainConfig.bech32Prefix &&\n (derivationMode === DerivationMode.NATIVE ||\n (derivationMode === DerivationMode.SEGWIT && !address.startsWith(\"1\")))\n );\n\n const verified = verify(\n proof.attestation,\n address,\n proof.proof,\n useSegwitEncoding,\n chainConfig\n );\n\n return {\n ...proof,\n status: verified ? ProofStatus.VERIFIED : ProofStatus.FAILED,\n };\n}\n\nfunction getDerivationMode(address: string) {\n if (address.match(\"^(bc1|tb1|ltc1).*\")) {\n return DerivationMode.NATIVE;\n } else if (address.match(\"^[32M].*\")) {\n return DerivationMode.SEGWIT;\n } else if (address.match(\"^[1nmL].*\")) {\n return DerivationMode.LEGACY;\n } else if (address.match(\"^(D).*\")) {\n return DerivationMode.DOGECOIN;\n } else if (address.match(\"^(q).*\")) {\n return DerivationMode.BCH;\n } else if (address.match(\"^(t1|t3).*\")) {\n return DerivationMode.LEGACY; // Zcash addresses\n } else if (address.match(\"^[X7].*\")) {\n return DerivationMode.LEGACY; // Dash addresses\n } else {\n throw new Error(\n \"INVALID ADDRESS: \"\n .concat(address)\n .concat(\" is not a valid or a supported address\")\n );\n }\n}\n\ntype DecodedSignature = {\n compressed: boolean;\n segwitType?: SEGWIT_TYPES;\n signature: SignatureType;\n};\n\nfunction decodeSignature(proof: string): DecodedSignature {\n const sigbytes = base64.decode(proof);\n if (sigbytes.length !== 65) throw new Error(\"Invalid signature length\");\n const flagByte = sigbytes[0] - 27;\n if (flagByte > 15 || flagByte < 0) {\n throw new Error(\"Invalid signature parameter\");\n }\n const compressed = !!(flagByte & 12); // Are there cases that aren't compressed?\n const recovery = flagByte & 3;\n const signature = secp256k1.Signature.fromCompact(sigbytes.slice(1));\n\n return {\n compressed,\n segwitType: !(flagByte & 8)\n ? undefined\n : !(flagByte & 4)\n ? SEGWIT_TYPES.P2SH_P2WPKH\n : SEGWIT_TYPES.P2WPKH,\n signature: signature.addRecoveryBit(recovery),\n };\n}\n\nfunction verify(\n attestation: string,\n address: string,\n proof: string,\n checkSegwitAlways: boolean,\n chainConfig: ChainConfig\n) {\n const { compressed, segwitType, signature } = decodeSignature(proof);\n if (checkSegwitAlways && !compressed) {\n throw new Error(\n \"checkSegwitAlways can only be used with a compressed pubkey signature flagbyte\"\n );\n }\n const hash = magicHash(attestation, chainConfig.messagePrefix);\n const publicKey = signature.recoverPublicKey(hash);\n const publicKeyBytes = publicKey.toRawBytes(compressed);\n const publicKeyHash = hash160(publicKeyBytes);\n let actual: string = \"\";\n\n // Special handling for Bitcoin Cash addresses\n if (address.startsWith(\"q\")) {\n // For BCH, we'll compare the public key hash directly since we're getting a CashAddr\n // Convert the CashAddr to legacy format for comparison\n actual = encodeBase58AddressFormat(\n chainConfig.pubKeyHashVersion,\n publicKeyHash\n );\n // Legacy P2PKH addresses in BCH start with '1' just like BTC\n // Source: https://reference.cash/protocol/blockchain/encoding/cashaddr#legacy-address-format\n return actual.startsWith(\"1\");\n }\n\n if (segwitType) {\n if (segwitType === SEGWIT_TYPES.P2SH_P2WPKH) {\n actual = encodeBase58AddressFormat(\n chainConfig.scriptHashVersion,\n publicKeyHash\n );\n } else {\n // parsed.segwitType === SEGWIT_TYPES.P2WPKH\n if (chainConfig.bech32Prefix) {\n actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);\n } else {\n // Fallback to legacy if bech32 not supported\n actual = encodeBase58AddressFormat(\n chainConfig.scriptHashVersion,\n publicKeyHash\n );\n // base58 can be p2pkh or p2sh-p2wpkh\n }\n }\n } else {\n // For addresses starting with \"3\" (P2SH), try both P2SH-P2WPKH and legacy P2SH encodings if segwitType is undefined\n if (address.startsWith(\"3\") && !segwitType) {\n // P2SH-P2WPKH: script hash of the redeem script (OP_0 <pubkeyhash>)\n const redeemScript = new Uint8Array(22);\n redeemScript[0] = 0x00; // OP_0\n redeemScript[1] = 0x14; // push 20 bytes\n redeemScript.set(publicKeyHash, 2);\n const redeemScriptHash = hash160(redeemScript);\n const p2shP2wpkh = encodeBase58AddressFormat(\n chainConfig.scriptHashVersion,\n redeemScriptHash\n );\n // Legacy P2SH: script hash of the public key\n const legacyP2sh = encodeBase58AddressFormat(\n chainConfig.scriptHashVersion,\n publicKeyHash\n );\n if (address === p2shP2wpkh || address === legacyP2sh) {\n return true;\n }\n actual = legacyP2sh; // fallback for error reporting\n } else if (address.startsWith(\"bc1q\") || address.startsWith(\"tb1q\") || address.startsWith(\"ltc1q\")) {\n // For native SegWit P2WPKH addresses (bc1q/tb1q/ltc1q), always encode as bech32\n // This handles Ledger wallets that sign without segwit flags\n if (chainConfig.bech32Prefix) {\n actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);\n } else {\n actual = encodeBase58AddressFormat(\n chainConfig.pubKeyHashVersion,\n publicKeyHash\n );\n }\n } else if (checkSegwitAlways && chainConfig.bech32Prefix) {\n try {\n actual = encodeBech32Address(publicKeyHash, chainConfig.bech32Prefix);\n // if address is bech32 it is not p2sh\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n } catch (e) {\n actual = encodeBase58AddressFormat(\n chainConfig.scriptHashVersion,\n publicKeyHash\n );\n // base58 can be p2pkh or p2sh-p2wpkh\n }\n } else {\n actual = encodeBase58AddressFormat(\n chainConfig.pubKeyHashVersion,\n publicKeyHash\n );\n }\n }\n\n return actual === address;\n}\n\nconst base58check = createBase58check(Hash.sha256);\n\nfunction encodeBase58AddressFormat(\n version: number | Uint8Array,\n publicKeyHash: Uint8Array\n) {\n const prefixBytes =\n typeof version === \"number\" ? Uint8Array.of(version) : version; // Accept raw Uint8Array for Zcash\n\n const payload = new Uint8Array(prefixBytes.length + publicKeyHash.length);\n payload.set(prefixBytes);\n payload.set(publicKeyHash, prefixBytes.length);\n return base58check.encode(payload);\n}\n\nfunction magicHash(attestation: string, messagePrefix: string) {\n const prefix = new TextEncoder().encode(messagePrefix);\n const message = new TextEncoder().encode(attestation);\n const length = encodeLength(message.length).buffer;\n const buffer = new Uint8Array(\n prefix.length + length.byteLength + message.length\n );\n buffer.set(prefix);\n buffer.set(new Uint8Array(length), prefix.length);\n buffer.set(message, prefix.length + length.byteLength);\n return hash256(buffer);\n}\n\nfunction encodeBech32Address(\n publicKeyHash: Uint8Array,\n prefix: string = \"bc\"\n): string {\n const bwords = bech32.toWords(publicKeyHash);\n bwords.unshift(0);\n return bech32.encode(prefix, bwords);\n}\n\nfunction hash256(buffer: Uint8Array): Uint8Array {\n return Hash.sha256(Hash.sha256(buffer));\n}\n\nfunction hash160(buffer: Uint8Array): Uint8Array {\n return Hash.ripemd160(Hash.sha256(buffer));\n}\n"]}