-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathhash-importer.ts
108 lines (103 loc) · 3.47 KB
/
hash-importer.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
/* global globalThis */
import { sha3512, keccak256 } from '@multiformats/sha3'
import { type Hasher, from } from 'multiformats/hashes/hasher'
import { identity } from 'multiformats/hashes/identity'
import * as sha2 from 'multiformats/hashes/sha2'
// #WhenAddingNewHasher
export type SupportedHashers = typeof sha2.sha256 |
typeof sha2.sha512 |
Hasher<'identity', 0x0> |
Hasher<'keccak-256', 27> |
Hasher<'sha1', 17> |
Hasher<'blake2b-256', 0xb220> |
Hasher<'blake2b-512', 0xb240> |
Hasher<'sha3-512', 20> |
Hasher<'blake3', 30>
export async function getHashersForCodes (...codes: number[]): Promise<SupportedHashers[]> {
const hashers: SupportedHashers[] = []
for (const code of codes) {
try {
hashers.push(await getHasherForCode(code))
} catch (error) {
console.error(`Failed to get hasher for code ${code}: ${error}`)
}
}
return hashers
}
/**
* Helper function to prevent `this` from being lost when calling `encode` and `digest` on a hasher.
*/
function getBoundHasher <T extends SupportedHashers> (hasher: T): T {
return {
...hasher,
encode: hasher.encode.bind(hasher),
digest: hasher.digest.bind(hasher)
}
}
export async function getHasherForCode (code: number): Promise<SupportedHashers> {
// #WhenAddingNewHasher
switch (code) {
case identity.code:
return getBoundHasher({
...identity,
name: 'identity' // multiformats/hashes/identity doesn't export a proper Hasher<Name, Code> type. See https://github.com/multiformats/js-multiformats/issues/313
})
case sha2.sha256.code:
return getBoundHasher(sha2.sha256)
case sha2.sha512.code:
return getBoundHasher(sha2.sha512)
case 17:
// git hasher uses sha1. see ipld-git/src/util.js
return getBoundHasher(from({
name: 'sha1',
code,
encode: async (data: Uint8Array): Promise<Uint8Array> => {
const hashBuffer = await globalThis.crypto.subtle.digest('SHA-1', data)
return new Uint8Array(hashBuffer)
}
}))
case sha3512.code: // sha3-512
return getBoundHasher(sha3512)
case keccak256.code: // keccak-256
return getBoundHasher(keccak256)
case 30:
return getBoundHasher(from({
name: 'blake3',
code,
encode: async (data: Uint8Array): Promise<Uint8Array> => {
const { createBLAKE3 } = await import('hash-wasm')
const blake3Hasher = await createBLAKE3()
blake3Hasher.init()
blake3Hasher.update(data)
return blake3Hasher.digest('binary')
}
}))
case 0xb220: // blake2b-256
return getBoundHasher(from({
name: 'blake2b-256',
code,
encode: async (data: Uint8Array): Promise<Uint8Array> => {
const { createBLAKE2b } = await import('hash-wasm')
const blake2bHasher = await createBLAKE2b(256)
blake2bHasher.init()
blake2bHasher.update(data)
return blake2bHasher.digest('binary')
}
}))
case 0xb240: // blake2b-512
return getBoundHasher(from({
name: 'blake2b-512',
code,
encode: async (data: Uint8Array): Promise<Uint8Array> => {
const { createBLAKE2b } = await import('hash-wasm')
const blake2bHasher = await createBLAKE2b(512)
blake2bHasher.init()
blake2bHasher.update(data)
return blake2bHasher.digest('binary')
}
})
)
default:
throw new Error(`unknown multihasher code '${code}'`)
}
}