Skip to content

Commit 3dee1dc

Browse files
authored
fix: use native crypto if available (#385)
* fix: use native crypto if available Adds the lodestar [crypto implementation](https://github.com/ChainSafe/lodestar/blob/6f27ac6e5f439577906cbfe5f9e01c59586c2af3/packages/beacon-node/src/network/libp2p/noise.ts#L3) to this module for use in node. I'm trying to get js-libp2p back onto the [libp2p performance dashboard](https://observablehq.com/@libp2p-workspace/performance-dashboard) and in testing this change increases streaming throughput from around 60 MB/s to almost 300 MB/s. * chore: fix electron main * chore: use subarray and also pass array length to buffer.concat
1 parent 1538267 commit 3dee1dc

File tree

4 files changed

+97
-5
lines changed

4 files changed

+97
-5
lines changed

package.json

+6-3
Original file line numberDiff line numberDiff line change
@@ -68,13 +68,15 @@
6868
"prepublish": "npm run build"
6969
},
7070
"dependencies": {
71+
"@chainsafe/as-chacha20poly1305": "^0.1.0",
72+
"@chainsafe/as-sha256": "^0.4.1",
7173
"@libp2p/crypto": "^2.0.0",
7274
"@libp2p/interface": "^0.1.0",
7375
"@libp2p/logger": "^3.0.0",
7476
"@libp2p/peer-id": "^3.0.0",
77+
"@noble/ciphers": "^0.4.0",
7578
"@noble/curves": "^1.1.0",
7679
"@noble/hashes": "^1.3.1",
77-
"@noble/ciphers": "^0.4.0",
7880
"it-byte-stream": "^1.0.0",
7981
"it-length-prefixed": "^9.0.1",
8082
"it-length-prefixed-stream": "^1.0.0",
@@ -83,7 +85,8 @@
8385
"it-stream-types": "^2.0.1",
8486
"protons-runtime": "^5.0.0",
8587
"uint8arraylist": "^2.4.3",
86-
"uint8arrays": "^4.0.4"
88+
"uint8arrays": "^4.0.4",
89+
"wherearewe": "^2.0.1"
8790
},
8891
"devDependencies": {
8992
"@chainsafe/libp2p-yamux": "^5.0.0",
@@ -108,7 +111,7 @@
108111
"sinon": "^16.1.3"
109112
},
110113
"browser": {
111-
"./dist/src/alloc-unsafe.js": "./dist/src/alloc-unsafe-browser.js",
114+
"./dist/src/crypto/index.js": "./dist/src/crypto/index.browser.js",
112115
"util": false
113116
}
114117
}

src/crypto/index.browser.ts

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
import { pureJsCrypto } from './js'
2+
3+
export const defaultCrypto = pureJsCrypto

src/crypto/index.ts

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
import crypto from 'node:crypto'
2+
import { newInstance, ChaCha20Poly1305 } from '@chainsafe/as-chacha20poly1305'
3+
import { digest } from '@chainsafe/as-sha256'
4+
import { isElectronMain } from 'wherearewe'
5+
import { pureJsCrypto } from './js.js'
6+
import type { ICryptoInterface } from '../crypto.js'
7+
8+
const ctx = newInstance()
9+
const asImpl = new ChaCha20Poly1305(ctx)
10+
const CHACHA_POLY1305 = 'chacha20-poly1305'
11+
const nodeCrypto: Pick<ICryptoInterface, 'hashSHA256' | 'chaCha20Poly1305Encrypt' | 'chaCha20Poly1305Decrypt'> = {
12+
hashSHA256 (data) {
13+
return crypto.createHash('sha256').update(data).digest()
14+
},
15+
16+
chaCha20Poly1305Encrypt (plaintext, nonce, ad, k) {
17+
const cipher = crypto.createCipheriv(CHACHA_POLY1305, k, nonce, {
18+
authTagLength: 16
19+
})
20+
cipher.setAAD(ad, { plaintextLength: plaintext.byteLength })
21+
const updated = cipher.update(plaintext)
22+
const final = cipher.final()
23+
const tag = cipher.getAuthTag()
24+
25+
const encrypted = Buffer.concat([updated, tag, final], updated.byteLength + tag.byteLength + final.byteLength)
26+
return encrypted
27+
},
28+
29+
chaCha20Poly1305Decrypt (ciphertext, nonce, ad, k, _dst) {
30+
const authTag = ciphertext.subarray(ciphertext.length - 16)
31+
const text = ciphertext.subarray(0, ciphertext.length - 16)
32+
const decipher = crypto.createDecipheriv(CHACHA_POLY1305, k, nonce, {
33+
authTagLength: 16
34+
})
35+
decipher.setAAD(ad, {
36+
plaintextLength: text.byteLength
37+
})
38+
decipher.setAuthTag(authTag)
39+
const updated = decipher.update(text)
40+
const final = decipher.final()
41+
if (final.byteLength > 0) {
42+
return Buffer.concat([updated, final], updated.byteLength + final.byteLength)
43+
}
44+
return updated
45+
}
46+
}
47+
48+
const asCrypto: Pick<ICryptoInterface, 'hashSHA256' | 'chaCha20Poly1305Encrypt' | 'chaCha20Poly1305Decrypt'> = {
49+
hashSHA256 (data) {
50+
return digest(data)
51+
},
52+
chaCha20Poly1305Encrypt (plaintext, nonce, ad, k) {
53+
return asImpl.seal(k, nonce, plaintext, ad)
54+
},
55+
chaCha20Poly1305Decrypt (ciphertext, nonce, ad, k, dst) {
56+
return asImpl.open(k, nonce, ciphertext, ad, dst)
57+
}
58+
}
59+
60+
// benchmarks show that for chacha20poly1305
61+
// the as implementation is faster for smaller payloads(<1200)
62+
// and the node implementation is faster for larger payloads
63+
export const defaultCrypto: ICryptoInterface = {
64+
...pureJsCrypto,
65+
hashSHA256 (data) {
66+
return nodeCrypto.hashSHA256(data)
67+
},
68+
chaCha20Poly1305Encrypt (plaintext, nonce, ad, k) {
69+
if (plaintext.length < 1200) {
70+
return asCrypto.chaCha20Poly1305Encrypt(plaintext, nonce, ad, k)
71+
}
72+
return nodeCrypto.chaCha20Poly1305Encrypt(plaintext, nonce, ad, k)
73+
},
74+
chaCha20Poly1305Decrypt (ciphertext, nonce, ad, k, dst) {
75+
if (ciphertext.length < 1200) {
76+
return asCrypto.chaCha20Poly1305Decrypt(ciphertext, nonce, ad, k, dst)
77+
}
78+
return nodeCrypto.chaCha20Poly1305Decrypt(ciphertext, nonce, ad, k, dst)
79+
}
80+
}
81+
82+
// no chacha20-poly1305 in electron https://github.com/electron/electron/issues/24024
83+
if (isElectronMain) {
84+
defaultCrypto.chaCha20Poly1305Encrypt = asCrypto.chaCha20Poly1305Encrypt
85+
defaultCrypto.chaCha20Poly1305Decrypt = asCrypto.chaCha20Poly1305Decrypt
86+
}

src/noise.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import { lpStream, type LengthPrefixedStream } from 'it-length-prefixed-stream'
33
import { duplexPair } from 'it-pair/duplex'
44
import { pipe } from 'it-pipe'
55
import { NOISE_MSG_MAX_LENGTH_BYTES } from './constants.js'
6-
import { pureJsCrypto } from './crypto/js.js'
6+
import { defaultCrypto } from './crypto/index.js'
77
import { decryptStream, encryptStream } from './crypto/streaming.js'
88
import { uint16BEDecode, uint16BEEncode } from './encoder.js'
99
import { XXHandshake } from './handshake-xx.js'
@@ -49,7 +49,7 @@ export class Noise implements INoiseConnection {
4949
constructor (init: NoiseInit = {}) {
5050
const { staticNoiseKey, extensions, crypto, prologueBytes, metrics } = init
5151

52-
this.crypto = crypto ?? pureJsCrypto
52+
this.crypto = crypto ?? defaultCrypto
5353
this.extensions = extensions
5454
this.metrics = metrics ? registerMetrics(metrics) : undefined
5555

0 commit comments

Comments
 (0)