From 7099987fa2ae3fd325db3e43854c0834b7156c34 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 1 Apr 2024 12:11:41 +1300 Subject: [PATCH 01/20] Add benchmarks package. --- package-lock.json | 15 ++++++++++++++ package.json | 2 +- packages/benchmarks/.aegir.js | 8 ++++++++ packages/benchmarks/README.md | 3 +++ packages/benchmarks/package.json | 34 +++++++++++++++++++++++++++++++ packages/benchmarks/tsconfig.json | 19 +++++++++++++++++ 6 files changed, 80 insertions(+), 1 deletion(-) create mode 100644 packages/benchmarks/.aegir.js create mode 100644 packages/benchmarks/README.md create mode 100644 packages/benchmarks/package.json create mode 100644 packages/benchmarks/tsconfig.json diff --git a/package-lock.json b/package-lock.json index dee223e7..3f5953fc 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3911,6 +3911,10 @@ "version": "2.2.0", "license": "MIT" }, + "node_modules/@organicdesign/db-benchamrks": { + "resolved": "packages/benchmarks", + "link": true + }, "node_modules/@organicdesign/db-cipher": { "resolved": "packages/cipher", "link": true @@ -30590,6 +30594,17 @@ "url": "https://github.com/sponsors/wooorm" } }, + "packages/benchmarks": { + "name": "@organicdesign/db-benchamrks", + "version": "0.1.0", + "license": "GPL-3.0-or-later", + "dependencies": { + "@organicdesign/db-client": "^0.1.0" + }, + "devDependencies": { + "aegir": "^42.2.4" + } + }, "packages/cipher": { "name": "@organicdesign/db-cipher", "version": "0.1.0", diff --git a/package.json b/package.json index 408024e1..08d0bc59 100644 --- a/package.json +++ b/package.json @@ -24,6 +24,6 @@ }, "private": true, "workspaces": [ - "packages/!(fuse)" + "packages/!(fuse|benchmarks)" ] } diff --git a/packages/benchmarks/.aegir.js b/packages/benchmarks/.aegir.js new file mode 100644 index 00000000..c3f0c16d --- /dev/null +++ b/packages/benchmarks/.aegir.js @@ -0,0 +1,8 @@ +export default { + build: { + config: { + platform: 'node', + format: 'esm' + } + } +} diff --git a/packages/benchmarks/README.md b/packages/benchmarks/README.md new file mode 100644 index 00000000..86a214fd --- /dev/null +++ b/packages/benchmarks/README.md @@ -0,0 +1,3 @@ +# Benchmarks + +Benchmarks for the distributed backup. diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json new file mode 100644 index 00000000..5d285bbd --- /dev/null +++ b/packages/benchmarks/package.json @@ -0,0 +1,34 @@ +{ + "type": "module", + "name": "@organicdesign/db-benchamrks", + "version": "0.1.0", + "description": "Benchmarks for the distributed backup.", + "main": "dist/src/index.js", + "types": "dist/src/index.d.ts", + "files": [ + "dist/src" + ], + "exports": { + ".": { + "types": "./dist/src/index.d.ts", + "import": "./dist/src/index.js" + } + }, + "scripts": { + "clean": "aegir clean", + "lint": "aegir lint", + "dep-check": "aegir dep-check", + "prepublishOnly": "npm run build", + "build": "aegir build" + }, + "author": "Saul Boyd", + "license": "GPL-3.0-or-later", + "devDependencies": { + "aegir": "^42.2.4" + }, + "private": true, + "sideEffects": false, + "dependencies": { + "@organicdesign/db-client": "^0.1.0" + } +} diff --git a/packages/benchmarks/tsconfig.json b/packages/benchmarks/tsconfig.json new file mode 100644 index 00000000..14e54efc --- /dev/null +++ b/packages/benchmarks/tsconfig.json @@ -0,0 +1,19 @@ +{ + "extends": "aegir/src/config/tsconfig.aegir.json", + "compilerOptions": { + "esModuleInterop": true, + "outDir": "dist", + "baseUrl": ".", + "module": "ES2022", + "target": "ES2022" + }, + "include": [ + "src", + "test" + ], + "references": [ + { + "path": "../client" + } + ] +} From 6541f1fec762c133d285a96e4a47900cfb1eaa99 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 1 Apr 2024 15:18:18 +1300 Subject: [PATCH 02/20] Add simple benchmark for 1kb transfer. --- package-lock.json | 3 + packages/benchmarks/package.json | 3 + packages/benchmarks/src/index.ts | 0 packages/benchmarks/src/transfer.ts | 58 +++++++++++++++ .../benchmarks/src/utils/generate-file.ts | 16 ++++ packages/benchmarks/src/utils/paths.ts | 6 ++ packages/benchmarks/src/utils/run-node.ts | 73 +++++++++++++++++++ packages/benchmarks/tsconfig.json | 3 + 8 files changed, 162 insertions(+) create mode 100644 packages/benchmarks/src/index.ts create mode 100644 packages/benchmarks/src/transfer.ts create mode 100644 packages/benchmarks/src/utils/generate-file.ts create mode 100644 packages/benchmarks/src/utils/paths.ts create mode 100644 packages/benchmarks/src/utils/run-node.ts diff --git a/package-lock.json b/package-lock.json index 3f5953fc..ed7efc8e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -30603,6 +30603,9 @@ }, "devDependencies": { "aegir": "^42.2.4" + }, + "peerDependencies": { + "@organicdesign/db-daemon": "^0.1.0" } }, "packages/cipher": { diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json index 5d285bbd..6f9bdff0 100644 --- a/packages/benchmarks/package.json +++ b/packages/benchmarks/package.json @@ -30,5 +30,8 @@ "sideEffects": false, "dependencies": { "@organicdesign/db-client": "^0.1.0" + }, + "peerDependencies": { + "@organicdesign/db-daemon": "^0.1.0" } } diff --git a/packages/benchmarks/src/index.ts b/packages/benchmarks/src/index.ts new file mode 100644 index 00000000..e69de29b diff --git a/packages/benchmarks/src/transfer.ts b/packages/benchmarks/src/transfer.ts new file mode 100644 index 00000000..4e1dbaeb --- /dev/null +++ b/packages/benchmarks/src/transfer.ts @@ -0,0 +1,58 @@ +import fs from 'fs/promises' +import Path from 'path' +import { Client } from '@organicdesign/db-client' +import generateFile from './utils/generate-file.js' +import { packagePath } from './utils/paths.js' +import runNode from './utils/run-node.js' + +const dataPath = Path.join(packagePath, 'test-out') +const name = 'transfer' + +const procs = await Promise.all([ + runNode(`${name}-0`), + runNode(`${name}-1`) +]) + +await Promise.all(procs.map(async p => p.start())) + +const clients = [ + new Client(Path.join(packagePath, `${name}-0.socket`)), + new Client(Path.join(packagePath, `${name}-1.socket`)) +] + +const addresses = await clients[0].addresses() + +await clients[1].connect(addresses[0]) + +const group = await clients[0].createGroup('test') + +await fs.mkdir(dataPath, { recursive: true }) + +const kbTestPath = Path.join(dataPath, '1kb.data') +await generateFile(kbTestPath, 2 ** 10) + +const [{ cid }] = await clients[0].import(group, kbTestPath) + +await clients[1].joinGroup(group) +await clients[1].sync() + +const start = Date.now() + +for (;;) { + const [{ state }] = await clients[1].getStatus([cid]) + + if (state === 'COMPLETED') { + break + } +} + +const time = Date.now() - start + +// eslint-disable-next-line no-console +console.log(`benchmark 1kb: ${time}ms`) + +for (const client of clients) { + client.stop() +} + +await Promise.all(procs.map(async p => p.stop())) diff --git a/packages/benchmarks/src/utils/generate-file.ts b/packages/benchmarks/src/utils/generate-file.ts new file mode 100644 index 00000000..31e74f2b --- /dev/null +++ b/packages/benchmarks/src/utils/generate-file.ts @@ -0,0 +1,16 @@ +import crypto from 'crypto' +import fs from 'fs' + +export default async (path: string, size: number, options?: { chunkSize: number }): Promise => { + const chunkSize = options?.chunkSize ?? 2 ** 14 + const stream = fs.createWriteStream(path) + + for (let i = 0; i < Math.ceil(size / chunkSize); i++) { + const thisSize = (i + 1) * chunkSize < size ? chunkSize : size - i * chunkSize + const data = crypto.randomBytes(thisSize) + + stream.write(data) + } + + await new Promise(resolve => stream.end(resolve)) +} diff --git a/packages/benchmarks/src/utils/paths.ts b/packages/benchmarks/src/utils/paths.ts new file mode 100644 index 00000000..7ab99848 --- /dev/null +++ b/packages/benchmarks/src/utils/paths.ts @@ -0,0 +1,6 @@ +import Path from 'path' +import { fileURLToPath } from 'url' + +export const packagePath = Path.join(Path.dirname(fileURLToPath(import.meta.url)), '../../../') + +export const modulesPath = Path.join(packagePath, '../../node_modules') diff --git a/packages/benchmarks/src/utils/run-node.ts b/packages/benchmarks/src/utils/run-node.ts new file mode 100644 index 00000000..3f45b115 --- /dev/null +++ b/packages/benchmarks/src/utils/run-node.ts @@ -0,0 +1,73 @@ +import { spawn, type ChildProcessWithoutNullStreams } from 'child_process' +import fs from 'fs/promises' +import Path from 'path' +import { DeferredPromise } from '@open-draft/deferred-promise' +import { toString as uint8ArrayToString } from 'uint8arrays' +import { packagePath, modulesPath } from './paths.js' + +export default async (name: string): Promise<{ start(): Promise, stop(): Promise }> => { + const socket = Path.join(packagePath, `${name}.socket`) + + const args = [ + Path.join(modulesPath, '@organicdesign/db-daemon/dist/src/index.js'), + '-s', socket + ] + + let proc: ChildProcessWithoutNullStreams + + const forceQuit = async (): Promise => { + // Ensure it is really dead. + proc.kill(9) + + await Promise.allSettled([ + fs.rm(socket) + ]) + } + + return { + async start () { + proc = spawn('node', args, { env: { ...process.env, DEBUG: 'backup:*' } }) + + const promise = new DeferredPromise() + + const listener = (chunk: Uint8Array): void => { + if (uint8ArrayToString(chunk).includes('started') === true) { + promise.resolve() + } + } + + proc.stderr.on('data', listener) + + await promise + + proc.stderr.off('data', listener) + }, + + async stop () { + const promise = new DeferredPromise() + + const listener = (chunk: Uint8Array): void => { + if (uint8ArrayToString(chunk).includes('exiting...') === true) { + promise.resolve() + } + } + + proc.stderr.on('data', listener) + proc.kill('SIGINT') + + // Kill it if it fails to do it cleanly. + setTimeout(() => { + forceQuit().finally(() => { + promise.reject(new Error('process did not exit cleanly')) + }) + }, 3000) + + await promise + + proc.stderr.off('data', listener) + + // Make sure things are cleaned up. + await forceQuit() + } + } +} diff --git a/packages/benchmarks/tsconfig.json b/packages/benchmarks/tsconfig.json index 14e54efc..90133985 100644 --- a/packages/benchmarks/tsconfig.json +++ b/packages/benchmarks/tsconfig.json @@ -14,6 +14,9 @@ "references": [ { "path": "../client" + }, + { + "path": "../daemon" } ] } From 6e50c6f1f82db2c3cdcb9ef26a7b1fbfca60ff0f Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 1 Apr 2024 17:20:59 +1300 Subject: [PATCH 03/20] Improve benchmark data. --- package-lock.json | 9 +- packages/benchmarks/package.json | 4 +- packages/benchmarks/src/transfer.ts | 58 ------------- packages/benchmarks/src/transfer/index.ts | 87 +++++++++++++++++++ packages/benchmarks/src/transfer/interface.ts | 5 ++ .../benchmarks/src/transfer/transfer-1kb.ts | 64 ++++++++++++++ 6 files changed, 167 insertions(+), 60 deletions(-) delete mode 100644 packages/benchmarks/src/transfer.ts create mode 100644 packages/benchmarks/src/transfer/index.ts create mode 100644 packages/benchmarks/src/transfer/interface.ts create mode 100644 packages/benchmarks/src/transfer/transfer-1kb.ts diff --git a/package-lock.json b/package-lock.json index ed7efc8e..edb5f51c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -28463,6 +28463,11 @@ "node": ">=14.0.0" } }, + "node_modules/tinybench": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.6.0.tgz", + "integrity": "sha512-N8hW3PG/3aOoZAN5V/NSAEDz0ZixDSSt5b/a05iqtpgfLWMSVuCo7w0k2vVvEjdrIoeGqZzweX2WlyioNIHchA==" + }, "node_modules/tmp": { "version": "0.0.33", "dev": true, @@ -30599,7 +30604,9 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { - "@organicdesign/db-client": "^0.1.0" + "@organicdesign/db-client": "^0.1.0", + "debug": "^4.3.4", + "tinybench": "^2.6.0" }, "devDependencies": { "aegir": "^42.2.4" diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json index 6f9bdff0..0fb42274 100644 --- a/packages/benchmarks/package.json +++ b/packages/benchmarks/package.json @@ -29,7 +29,9 @@ "private": true, "sideEffects": false, "dependencies": { - "@organicdesign/db-client": "^0.1.0" + "@organicdesign/db-client": "^0.1.0", + "debug": "^4.3.4", + "tinybench": "^2.6.0" }, "peerDependencies": { "@organicdesign/db-daemon": "^0.1.0" diff --git a/packages/benchmarks/src/transfer.ts b/packages/benchmarks/src/transfer.ts deleted file mode 100644 index 4e1dbaeb..00000000 --- a/packages/benchmarks/src/transfer.ts +++ /dev/null @@ -1,58 +0,0 @@ -import fs from 'fs/promises' -import Path from 'path' -import { Client } from '@organicdesign/db-client' -import generateFile from './utils/generate-file.js' -import { packagePath } from './utils/paths.js' -import runNode from './utils/run-node.js' - -const dataPath = Path.join(packagePath, 'test-out') -const name = 'transfer' - -const procs = await Promise.all([ - runNode(`${name}-0`), - runNode(`${name}-1`) -]) - -await Promise.all(procs.map(async p => p.start())) - -const clients = [ - new Client(Path.join(packagePath, `${name}-0.socket`)), - new Client(Path.join(packagePath, `${name}-1.socket`)) -] - -const addresses = await clients[0].addresses() - -await clients[1].connect(addresses[0]) - -const group = await clients[0].createGroup('test') - -await fs.mkdir(dataPath, { recursive: true }) - -const kbTestPath = Path.join(dataPath, '1kb.data') -await generateFile(kbTestPath, 2 ** 10) - -const [{ cid }] = await clients[0].import(group, kbTestPath) - -await clients[1].joinGroup(group) -await clients[1].sync() - -const start = Date.now() - -for (;;) { - const [{ state }] = await clients[1].getStatus([cid]) - - if (state === 'COMPLETED') { - break - } -} - -const time = Date.now() - start - -// eslint-disable-next-line no-console -console.log(`benchmark 1kb: ${time}ms`) - -for (const client of clients) { - client.stop() -} - -await Promise.all(procs.map(async p => p.stop())) diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts new file mode 100644 index 00000000..5706aa46 --- /dev/null +++ b/packages/benchmarks/src/transfer/index.ts @@ -0,0 +1,87 @@ +/* eslint-disable no-console,no-loop-func */ +import debug from 'debug' +import { Bench } from 'tinybench' +import { createTransferBench } from './transfer-1kb.js' +import type { TransferBenchmark } from './interface.js' +import prettyBytes from 'pretty-bytes'; + +const log = debug('bench:transfer') + +const ITERATIONS = parseInt(process.env.ITERATIONS ?? '3') +const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') +const RESULT_PRECISION = 2 + +const zeros = [ + 0, // 1b + 3, // 1kb + 6, // 1mb + 8, // 100mb +] + +const sizes = zeros.map(z => 10 ** z) + +const impls: Array<{ + name: string + create(): Promise + results: number[] } +> = sizes.map(size => ({ + name: `${prettyBytes(size)}`, + create: async () => { + const bench = await createTransferBench(size) + + return { ...bench, size } + }, + results: [] +})) + +async function main (): Promise { + const suite = new Bench({ + iterations: ITERATIONS, + time: MIN_TIME + }) + + for (const impl of impls) { + const run = async function (): Promise { + log('Start: setup') + const subject = await impl.create() + log('End: setup') + + await subject.warmup() + const start = performance.now() + await subject.run() + impl.results.push(performance.now() - start) + + log('Start: teardown') + await subject.teardown() + log('End: teardown') + } + + const hooks = { + beforeEach: async () => { + log(`Start: test ${impl.name}`) + }, + afterEach: async () => { + log(`End: test ${impl.name}`) + } + } + + suite.add(impl.name, run, hooks) + } + + await suite.run() + + console.table(suite.tasks.map(({ name, result }) => { + return { + 'File Size': name, + 'runs/s': result?.hz.toFixed(RESULT_PRECISION), + 'ms/run': result?.period.toFixed(RESULT_PRECISION), + runs: result?.samples.length, + p99: result?.p99.toFixed(RESULT_PRECISION) + } + })) +} + +main().catch(err => { + console.error(err) // eslint-disable-line no-console + process.exit(1) +}) diff --git a/packages/benchmarks/src/transfer/interface.ts b/packages/benchmarks/src/transfer/interface.ts new file mode 100644 index 00000000..90fc6c23 --- /dev/null +++ b/packages/benchmarks/src/transfer/interface.ts @@ -0,0 +1,5 @@ +export interface TransferBenchmark { + teardown(): Promise + run(): Promise + warmup(): Promise +} diff --git a/packages/benchmarks/src/transfer/transfer-1kb.ts b/packages/benchmarks/src/transfer/transfer-1kb.ts new file mode 100644 index 00000000..3cbe5710 --- /dev/null +++ b/packages/benchmarks/src/transfer/transfer-1kb.ts @@ -0,0 +1,64 @@ +import fs from 'fs/promises' +import Path from 'path' +import { Client } from '@organicdesign/db-client' +import generateFile from '../utils/generate-file.js' +import { packagePath } from '../utils/paths.js' +import runNode from '../utils/run-node.js' +import type { TransferBenchmark } from './interface.js' + +export const createTransferBench = async (size: number): Promise => { + const dataPath = Path.join(packagePath, 'test-out') + const name = 'transfer' + + const procs = await Promise.all([ + runNode(`${name}-0`), + runNode(`${name}-1`) + ]) + + await Promise.all(procs.map(async p => p.start())) + + const clients = [ + new Client(Path.join(packagePath, `${name}-0.socket`)), + new Client(Path.join(packagePath, `${name}-1.socket`)) + ] + + const addresses = await clients[0].addresses() + + await clients[1].connect(addresses[0]) + + const group = await clients[0].createGroup('test') + + await fs.mkdir(dataPath, { recursive: true }) + + const kbTestPath = Path.join(dataPath, '1kb.data') + await generateFile(kbTestPath, size) + + const [{ cid }] = await clients[0].import(group, kbTestPath) + + return { + async teardown () { + for (const client of clients) { + client.stop() + } + + await Promise.all(procs.map(async p => p.stop())) + }, + + async warmup () { + await clients[1].joinGroup(group) + await clients[1].sync() + }, + + async run () { + for (;;) { + const [{ state }] = await clients[1].getStatus([cid]) + + if (state === 'COMPLETED') { + break + } + + await new Promise(resolve => setTimeout(resolve, 10)) + } + } + } +} From 95a0fa1fbbfb677fbaabe9c276a86ced288c901e Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 1 Apr 2024 17:30:12 +1300 Subject: [PATCH 04/20] Cleanup benchmark. --- package-lock.json | 12 ++++++++++ packages/benchmarks/package.json | 1 + ...ansfer-1kb.ts => create-transfer-bench.ts} | 24 +++++++++---------- packages/benchmarks/src/transfer/index.ts | 11 ++++----- 4 files changed, 29 insertions(+), 19 deletions(-) rename packages/benchmarks/src/transfer/{transfer-1kb.ts => create-transfer-bench.ts} (70%) diff --git a/package-lock.json b/package-lock.json index edb5f51c..8103a864 100644 --- a/package-lock.json +++ b/package-lock.json @@ -25095,6 +25095,17 @@ "node": ">=6" } }, + "node_modules/pretty-bytes": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-6.1.1.tgz", + "integrity": "sha512-mQUvGU6aUFQ+rNvTIAcZuWGRT9a6f6Yrg9bHs4ImKF+HZCEK+plBvnAZYSIQztknZF2qnzNtr6F8s0+IuptdlQ==", + "engines": { + "node": "^14.13.1 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/pretty-format": { "version": "26.6.2", "license": "MIT", @@ -30606,6 +30617,7 @@ "dependencies": { "@organicdesign/db-client": "^0.1.0", "debug": "^4.3.4", + "pretty-bytes": "^6.1.1", "tinybench": "^2.6.0" }, "devDependencies": { diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json index 0fb42274..3583f5c3 100644 --- a/packages/benchmarks/package.json +++ b/packages/benchmarks/package.json @@ -31,6 +31,7 @@ "dependencies": { "@organicdesign/db-client": "^0.1.0", "debug": "^4.3.4", + "pretty-bytes": "^6.1.1", "tinybench": "^2.6.0" }, "peerDependencies": { diff --git a/packages/benchmarks/src/transfer/transfer-1kb.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts similarity index 70% rename from packages/benchmarks/src/transfer/transfer-1kb.ts rename to packages/benchmarks/src/transfer/create-transfer-bench.ts index 3cbe5710..666a7a73 100644 --- a/packages/benchmarks/src/transfer/transfer-1kb.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -8,19 +8,14 @@ import type { TransferBenchmark } from './interface.js' export const createTransferBench = async (size: number): Promise => { const dataPath = Path.join(packagePath, 'test-out') - const name = 'transfer' - const procs = await Promise.all([ - runNode(`${name}-0`), - runNode(`${name}-1`) - ]) + const names = [...Array(2).keys()].map(i => `transfer-${size}-${i}`) + + const procs = await Promise.all(names.map(async n => runNode(n))) await Promise.all(procs.map(async p => p.start())) - const clients = [ - new Client(Path.join(packagePath, `${name}-0.socket`)), - new Client(Path.join(packagePath, `${name}-1.socket`)) - ] + const clients = names.map(n => new Client(Path.join(packagePath, `${n}.socket`))) const addresses = await clients[0].addresses() @@ -30,10 +25,10 @@ export const createTransferBench = async (size: number): Promise p.stop())) + await Promise.all([ + ...procs.map(async p => p.stop()), + fs.rm(dataFile) + ]) }, async warmup () { diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index 5706aa46..1c2b085c 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -1,9 +1,9 @@ /* eslint-disable no-console,no-loop-func */ import debug from 'debug' +import prettyBytes from 'pretty-bytes' import { Bench } from 'tinybench' -import { createTransferBench } from './transfer-1kb.js' +import { createTransferBench } from './create-transfer-bench.js' import type { TransferBenchmark } from './interface.js' -import prettyBytes from 'pretty-bytes'; const log = debug('bench:transfer') @@ -12,10 +12,9 @@ const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') const RESULT_PRECISION = 2 const zeros = [ - 0, // 1b - 3, // 1kb - 6, // 1mb - 8, // 100mb + 0, // 1b + 3, // 1kb + 6 // 1mb ] const sizes = zeros.map(z => 10 ** z) From f908b2591f7e3b973a62c3decaaa90ea21aeafd6 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 1 Apr 2024 19:39:07 +1300 Subject: [PATCH 05/20] Reduce file creations. --- .../src/transfer/create-transfer-bench.ts | 7 +--- packages/benchmarks/src/transfer/index.ts | 39 ++++++++++++++++--- 2 files changed, 35 insertions(+), 11 deletions(-) diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts index 666a7a73..7ac8bec2 100644 --- a/packages/benchmarks/src/transfer/create-transfer-bench.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -1,7 +1,6 @@ import fs from 'fs/promises' import Path from 'path' import { Client } from '@organicdesign/db-client' -import generateFile from '../utils/generate-file.js' import { packagePath } from '../utils/paths.js' import runNode from '../utils/run-node.js' import type { TransferBenchmark } from './interface.js' @@ -26,7 +25,6 @@ export const createTransferBench = async (size: number): Promise p.stop()), - fs.rm(dataFile) - ]) + await Promise.all(procs.map(async p => p.stop())) }, async warmup () { diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index 1c2b085c..feef6cac 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -1,7 +1,11 @@ /* eslint-disable no-console,no-loop-func */ +import fs from 'fs/promises' +import Path from 'path' import debug from 'debug' import prettyBytes from 'pretty-bytes' import { Bench } from 'tinybench' +import generateFile from '../utils/generate-file.js' +import { packagePath } from '../utils/paths.js' import { createTransferBench } from './create-transfer-bench.js' import type { TransferBenchmark } from './interface.js' @@ -14,7 +18,7 @@ const RESULT_PRECISION = 2 const zeros = [ 0, // 1b 3, // 1kb - 6 // 1mb + 6 // 1mb ] const sizes = zeros.map(z => 10 ** z) @@ -22,21 +26,46 @@ const sizes = zeros.map(z => 10 ** z) const impls: Array<{ name: string create(): Promise - results: number[] } -> = sizes.map(size => ({ + results: number[] + size: number +}> = sizes.map(size => ({ name: `${prettyBytes(size)}`, create: async () => { const bench = await createTransferBench(size) return { ...bench, size } }, - results: [] + results: [], + size })) +const dataPath = Path.join(packagePath, 'test-out') + async function main (): Promise { const suite = new Bench({ iterations: ITERATIONS, - time: MIN_TIME + time: MIN_TIME, + async setup (task) { + const impl = impls.find(({ name }) => task.name.includes(name)) + + if (impl == null) { + return + } + + const dataFile = Path.join(dataPath, `${impl.size}.data`) + await generateFile(dataFile, impl.size) + }, + + async teardown (task) { + const impl = impls.find(({ name }) => task.name.includes(name)) + + if (impl == null) { + return + } + + const dataFile = Path.join(dataPath, `${impl.size}.data`) + await fs.rm(dataFile) + } }) for (const impl of impls) { From 1b8c2887c94e4c49bcda46eceea5d42fc8783b27 Mon Sep 17 00:00:00 2001 From: saul Date: Tue, 2 Apr 2024 13:07:39 +1300 Subject: [PATCH 06/20] Add speeds to benchmarks. --- .../src/transfer/create-transfer-bench.ts | 4 ++ packages/benchmarks/src/transfer/index.ts | 51 +++++++++++-------- packages/benchmarks/src/transfer/interface.ts | 11 ++++ 3 files changed, 46 insertions(+), 20 deletions(-) diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts index 7ac8bec2..3792e034 100644 --- a/packages/benchmarks/src/transfer/create-transfer-bench.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -27,8 +27,12 @@ export const createTransferBench = async (size: number): Promise 10 ** z) -const impls: Array<{ - name: string - create(): Promise - results: number[] - size: number -}> = sizes.map(size => ({ +const impls: TransferImplementation[] = sizes.map(size => ({ name: `${prettyBytes(size)}`, - create: async () => { - const bench = await createTransferBench(size) - - return { ...bench, size } - }, + create: async () => createTransferBench(size), results: [], - size + fileSize: size, + size: 0, + blocks: 0 })) const dataPath = Path.join(packagePath, 'test-out') @@ -45,6 +40,7 @@ async function main (): Promise { const suite = new Bench({ iterations: ITERATIONS, time: MIN_TIME, + async setup (task) { const impl = impls.find(({ name }) => task.name.includes(name)) @@ -52,8 +48,8 @@ async function main (): Promise { return } - const dataFile = Path.join(dataPath, `${impl.size}.data`) - await generateFile(dataFile, impl.size) + const dataFile = Path.join(dataPath, `${impl.fileSize}.data`) + await generateFile(dataFile, impl.fileSize) }, async teardown (task) { @@ -63,7 +59,7 @@ async function main (): Promise { return } - const dataFile = Path.join(dataPath, `${impl.size}.data`) + const dataFile = Path.join(dataPath, `${impl.fileSize}.data`) await fs.rm(dataFile) } }) @@ -77,6 +73,9 @@ async function main (): Promise { await subject.warmup() const start = performance.now() await subject.run() + + impl.size = subject.size + impl.blocks = subject.blocks impl.results.push(performance.now() - start) log('Start: teardown') @@ -99,11 +98,23 @@ async function main (): Promise { await suite.run() console.table(suite.tasks.map(({ name, result }) => { + const impl = impls.find(impl => impl.name === name) + + if (impl == null) { + throw new Error('got result without implementation') + } + + const speed = impl.size / (result?.period ?? 0 / 1000) + const bps = impl.blocks / (result?.period ?? 0 / 1000) + return { 'File Size': name, - 'runs/s': result?.hz.toFixed(RESULT_PRECISION), - 'ms/run': result?.period.toFixed(RESULT_PRECISION), - runs: result?.samples.length, + Size: prettyBytes(impl.size), + Blocks: impl.blocks, + 'Speed (Size)': `${prettyBytes(speed)}/s`, + 'Speed (Blocks)': `${(bps).toFixed(RESULT_PRECISION)} blocks/s`, + 'Run Time': result?.period.toFixed(RESULT_PRECISION), + Runs: result?.samples.length, p99: result?.p99.toFixed(RESULT_PRECISION) } })) diff --git a/packages/benchmarks/src/transfer/interface.ts b/packages/benchmarks/src/transfer/interface.ts index 90fc6c23..33493a07 100644 --- a/packages/benchmarks/src/transfer/interface.ts +++ b/packages/benchmarks/src/transfer/interface.ts @@ -1,5 +1,16 @@ export interface TransferBenchmark { + blocks: number + size: number teardown(): Promise run(): Promise warmup(): Promise } + +export interface TransferImplementation { + name: string + create(): Promise + results: number[] + fileSize: number + size: number + blocks: number +} From 6ee118ac86d1f82a25ea8c0f65ea4d038a68189b Mon Sep 17 00:00:00 2001 From: saul Date: Tue, 2 Apr 2024 13:13:40 +1300 Subject: [PATCH 07/20] Add benchmarks info to readme. --- README.md | 28 ++++++++++++++++++++++- packages/benchmarks/src/transfer/index.ts | 3 ++- 2 files changed, 29 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index aeed5ab9..f2ba5a96 100644 --- a/README.md +++ b/README.md @@ -27,7 +27,7 @@ You can enable fuse by adding it to the `workspaces` value in `package.json`: ```json "workspaces": [ - "packages/!(fuse)", + "packages/!(fuse|benchmarks)", "packages/fuse" ] ``` @@ -191,3 +191,29 @@ node packages/cli/dist/src/index.js list-groups ``` node packages/cli/dist/src/index.js join-group $GROUP ``` + +## Benchmarks + +You can enable benchmarks by adding it to the `workspaces` value in `package.json`: + +```json +"workspaces": [ + "packages/!(fuse|benchmarks)", + "packages/benchmarks" +] +``` + +Then you can build the benchmarks package as usual: + +``` +npm ci +npm run build +``` + +### Transfer + +To run the transfer benchmark: + +``` +node packages/benchmarks/dist/src/transfer/index.js +``` diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index e6b24f33..a63a4c2f 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -20,7 +20,8 @@ const zeros = [ 3, // 1kb 6, // 1mb 7, // 10mb - 8 // 100mb + 8, // 100mb + 9 // 1gb ] const sizes = zeros.map(z => 10 ** z) From b39d59efc0575e161f4b38996e95bcc6b41cd621 Mon Sep 17 00:00:00 2001 From: saul Date: Thu, 4 Apr 2024 11:21:51 +1300 Subject: [PATCH 08/20] Fix benchmarks calculation. --- packages/benchmarks/src/transfer/index.ts | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index a63a4c2f..7d4904f7 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -15,16 +15,14 @@ const ITERATIONS = parseInt(process.env.ITERATIONS ?? '3') const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') const RESULT_PRECISION = 2 -const zeros = [ +const sizes = [ 0, // 1b 3, // 1kb 6, // 1mb 7, // 10mb 8, // 100mb 9 // 1gb -] - -const sizes = zeros.map(z => 10 ** z) +].map(z => 10 ** z) const impls: TransferImplementation[] = sizes.map(size => ({ name: `${prettyBytes(size)}`, @@ -105,18 +103,19 @@ async function main (): Promise { throw new Error('got result without implementation') } - const speed = impl.size / (result?.period ?? 0 / 1000) - const bps = impl.blocks / (result?.period ?? 0 / 1000) + const seconds = (result?.period ?? 0) / 1000 + const speed = impl.size / seconds + const bps = impl.blocks / seconds return { 'File Size': name, Size: prettyBytes(impl.size), Blocks: impl.blocks, 'Speed (Size)': `${prettyBytes(speed)}/s`, - 'Speed (Blocks)': `${(bps).toFixed(RESULT_PRECISION)} blocks/s`, - 'Run Time': result?.period.toFixed(RESULT_PRECISION), + 'Speed (Blocks)': `${bps.toFixed(RESULT_PRECISION)} blocks/s`, + 'Run Time': `${result?.period.toFixed(RESULT_PRECISION)}ms`, Runs: result?.samples.length, - p99: result?.p99.toFixed(RESULT_PRECISION) + p99: `${result?.p99.toFixed(RESULT_PRECISION)}ms` } })) } From e360723dd283dc61e8e412b5e4802c8e167c56f6 Mon Sep 17 00:00:00 2001 From: saul Date: Thu, 4 Apr 2024 11:46:31 +1300 Subject: [PATCH 09/20] Fix benchmark file write stream. --- packages/benchmarks/src/utils/generate-file.ts | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/packages/benchmarks/src/utils/generate-file.ts b/packages/benchmarks/src/utils/generate-file.ts index 31e74f2b..e92d75f1 100644 --- a/packages/benchmarks/src/utils/generate-file.ts +++ b/packages/benchmarks/src/utils/generate-file.ts @@ -9,7 +9,15 @@ export default async (path: string, size: number, options?: { chunkSize: number const thisSize = (i + 1) * chunkSize < size ? chunkSize : size - i * chunkSize const data = crypto.randomBytes(thisSize) - stream.write(data) + await new Promise((resolve, reject) => { + stream.write(data, (error) => { + if (error != null) { + reject(error) + } else { + resolve(error) + } + }) + }) } await new Promise(resolve => stream.end(resolve)) From fc153ec0519a4158c9747293c9d4836689a657dc Mon Sep 17 00:00:00 2001 From: saul Date: Thu, 4 Apr 2024 13:21:40 +1300 Subject: [PATCH 10/20] Fix benchmark folder creation. --- packages/benchmarks/src/transfer/create-transfer-bench.ts | 6 +----- packages/benchmarks/src/transfer/index.ts | 3 +++ 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts index 3792e034..74540c76 100644 --- a/packages/benchmarks/src/transfer/create-transfer-bench.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -1,4 +1,3 @@ -import fs from 'fs/promises' import Path from 'path' import { Client } from '@organicdesign/db-client' import { packagePath } from '../utils/paths.js' @@ -14,7 +13,7 @@ export const createTransferBench = async (size: number): Promise p.start())) - const clients = names.map(n => new Client(Path.join(packagePath, `${n}.socket`))) + const clients = names.map(n => new Client(Path.join(dataPath, n, `socket`))) const addresses = await clients[0].addresses() @@ -22,10 +21,7 @@ export const createTransferBench = async (size: number): Promise { return } + await fs.mkdir(dataPath, { recursive: true }) const dataFile = Path.join(dataPath, `${impl.fileSize}.data`) await generateFile(dataFile, impl.fileSize) }, @@ -118,6 +119,8 @@ async function main (): Promise { p99: `${result?.p99.toFixed(RESULT_PRECISION)}ms` } })) + + await fs.rm(dataPath, { recursive: true }) } main().catch(err => { From 7a42234ec4aedcce78ce68fdac1ce17d29e39fdf Mon Sep 17 00:00:00 2001 From: saul Date: Thu, 4 Apr 2024 14:32:04 +1300 Subject: [PATCH 11/20] Fix memory hang issue. --- packages/utils/src/dag/index.ts | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/packages/utils/src/dag/index.ts b/packages/utils/src/dag/index.ts index b1b3b2fd..bf34b675 100644 --- a/packages/utils/src/dag/index.ts +++ b/packages/utils/src/dag/index.ts @@ -70,10 +70,14 @@ export const getSize = async (blockstore: Blockstore, cid: CID): Promise<{ block let blocks = 0 for await (const getBlock of walk(blockstore, cid)) { - const { block } = await getBlock() + const data = await getBlock() blocks++ - size += block.length + size += data.block.length + + // This is needed to signal to NodeJS to free up the memory imediately, + // otherwise the memory can hang around for a while and can be massive. + delete (data as { block?: Uint8Array }).block } return { size, blocks } From 55b7b274d77b93fbbd31fe70af2c348f25f7c61f Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 15:43:00 +1200 Subject: [PATCH 12/20] Add local option to walk dag. --- packages/utils/src/dag/index.ts | 12 ++++++-- packages/utils/test/dag-size.spec.ts | 27 +----------------- packages/utils/test/walk.spec.ts | 42 ++++++++++++++++++++++++++++ 3 files changed, 53 insertions(+), 28 deletions(-) create mode 100644 packages/utils/test/walk.spec.ts diff --git a/packages/utils/src/dag/index.ts b/packages/utils/src/dag/index.ts index bf34b675..d518af66 100644 --- a/packages/utils/src/dag/index.ts +++ b/packages/utils/src/dag/index.ts @@ -20,7 +20,7 @@ export const getWalker = (cid: CID): DAGWalker => { return dagWalker } -export const walk = async function * (blockstore: Blockstore, cid: CID, maxDepth?: number, options?: AbortOptions): AsyncGenerator<() => Promise> { +export const walk = async function * (blockstore: Blockstore, cid: CID, options: AbortOptions & { local?: boolean, maxDepth?: number } = {}): AsyncGenerator<() => Promise> { const queue: Array<() => Promise> = [] const promises: Array> = [] @@ -33,9 +33,17 @@ export const walk = async function * (blockstore: Blockstore, cid: CID, maxDepth throw new Error(`No dag walker found for cid codec ${cid.code}`) } + if (options.local === true) { + const has = await blockstore.has(cid, options) + + if (!has) { + throw new Error(`missing block ${cid.toString()}`) + } + } + const block = await blockstore.get(cid, options) - if (maxDepth == null || depth < maxDepth) { + if (options.maxDepth == null || depth < options.maxDepth) { for await (const cid of dagWalker.walk(block)) { enqueue(cid, depth + 1) } diff --git a/packages/utils/test/dag-size.spec.ts b/packages/utils/test/dag-size.spec.ts index 60373486..8c9f878e 100644 --- a/packages/utils/test/dag-size.spec.ts +++ b/packages/utils/test/dag-size.spec.ts @@ -2,7 +2,7 @@ import assert from 'assert/strict' import { createDag } from '@organicdesign/db-test-utils' import { MemoryBlockstore } from 'blockstore-core' import { type CID } from 'multiformats/cid' -import { getSize, walk } from '../src/dag/index.js' +import { getSize } from '../src/dag/index.js' describe('getDagSize', () => { let dag: CID[] @@ -28,28 +28,3 @@ describe('getDagSize', () => { assert.equal(size, totalSize) }) }) - -describe('walkDag', () => { - let dag: CID[] - let blockstore: MemoryBlockstore - - before(async () => { - blockstore = new MemoryBlockstore() - - dag = await createDag({ blockstore }, 3, 3) - }) - - it('walks over every value of the dag', async () => { - let count = 0 - - for await (const getData of walk(blockstore, dag[0])) { - const data = await getData() - - assert(dag.find(cid => cid.equals(data.cid))) - - count++ - } - - assert.equal(count, dag.length) - }) -}) diff --git a/packages/utils/test/walk.spec.ts b/packages/utils/test/walk.spec.ts new file mode 100644 index 00000000..d4cc931d --- /dev/null +++ b/packages/utils/test/walk.spec.ts @@ -0,0 +1,42 @@ +import assert from 'assert/strict' +import { createDag } from '@organicdesign/db-test-utils' +import { MemoryBlockstore } from 'blockstore-core' +import all from 'it-all' +import parallel from 'it-parallel' +import { pipe } from 'it-pipe' +import { type CID } from 'multiformats/cid' +import { walk } from '../src/dag/index.js' + +describe('walkDag', () => { + let dag: CID[] + let blockstore: MemoryBlockstore + + before(async () => { + blockstore = new MemoryBlockstore() + + dag = await createDag({ blockstore }, 3, 3) + }) + + it('walks over every value of the dag', async () => { + let count = 0 + + for await (const getData of walk(blockstore, dag[0])) { + const data = await getData() + + assert(dag.find(cid => cid.equals(data.cid))) + + count++ + } + + assert.equal(count, dag.length) + }) + + it('throws an error if the blockstore is missing an item and local is set', async () => { + const partialBlockstore = new MemoryBlockstore() + const dag = await createDag({ blockstore: partialBlockstore }, 3, 3) + + await partialBlockstore.delete(dag[1]) + + await assert.rejects(async () => all(pipe(walk(partialBlockstore, dag[0], { local: true }), parallel))) + }) +}) From f00410fb92db9884192bdd35b15d8d150c2a8d8b Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 15:47:21 +1200 Subject: [PATCH 13/20] Remove block from helia pin manager results and use dag walker. --- packages/helia-pin-manager/src/pin-manager.ts | 39 +++++++------------ 1 file changed, 14 insertions(+), 25 deletions(-) diff --git a/packages/helia-pin-manager/src/pin-manager.ts b/packages/helia-pin-manager/src/pin-manager.ts index b5b36859..b523887e 100644 --- a/packages/helia-pin-manager/src/pin-manager.ts +++ b/packages/helia-pin-manager/src/pin-manager.ts @@ -1,5 +1,5 @@ import { DeferredPromise } from '@open-draft/deferred-promise' -import { getWalker } from '@organicdesign/db-utils/dag' +import { walk, getWalker } from '@organicdesign/db-utils/dag' import { NamespaceDatastore } from 'datastore-core' import { Key, type Datastore } from 'interface-datastore' import all from 'it-all' @@ -17,7 +17,6 @@ export interface Components { } export interface BlockInfo { - block: Uint8Array cid: CID links: CID[] } @@ -36,7 +35,7 @@ class CIDEvent extends Event { export class PinManager { readonly events = new EventTarget<[CIDEvent]>() private readonly helia: Helia - private readonly activeDownloads = new Map>() + private readonly activeDownloads = new Map>() private readonly pins: Pins private readonly blocks: Blocks private readonly downloads: Downloads @@ -96,30 +95,20 @@ export class PinManager { throw new Error('pin find or create failed') } - const walk = async (subCid: CID, depth: number): Promise => { - const dagWalker = getWalker(subCid) + for await (const getBlock of walk(this.helia.blockstore, cid, { local: true })) { + const data = await getBlock() - if (!await this.helia.blockstore.has(subCid)) { - throw new Error('pin does not exist locally') - } - - await addBlockRef(this.helia, subCid, cid) + await addBlockRef(this.helia, data.cid, cid) - const block = await this.helia.blockstore.get(subCid) - - await this.blocks.getOrPut(cid, subCid, { - size: block.length, - depth, + await this.blocks.getOrPut(cid, data.cid, { + size: data.block.length, + depth: data.depth, timestamp: Date.now() }) - for await (const cid of dagWalker.walk(block)) { - await walk(cid, depth + 1) - } + delete (data as { block?: Uint8Array }).block } - await walk(cid, 0) - await addPinRef(this.helia, cid) await this.pins.put(cid, { @@ -285,7 +274,7 @@ export class PinManager { const enqueue = (cid: CID, depth: number): void => { queue.push(async () => { - const { links, block } = await this.download(cid) + const { links } = await this.download(cid) if (pinData.depth == null || depth < pinData.depth) { for (const cid of links) { @@ -293,7 +282,7 @@ export class PinManager { } } - return { cid, block, links } + return { cid, links } }) } @@ -303,7 +292,7 @@ export class PinManager { enqueue(head.cid, head.depth) } - const promises: Array> = [] + const promises: Array> = [] while (queue.length + promises.length !== 0) { const func = queue.shift() @@ -314,7 +303,7 @@ export class PinManager { continue } - const promise = new DeferredPromise<{ cid: CID, block: Uint8Array }>() + const promise = new DeferredPromise<{ cid: CID }>() promises.push(promise) @@ -393,7 +382,7 @@ export class PinManager { this.events.dispatchEvent(new CIDEvent('downloads:added', cid)) - return { block, cid, links } + return { cid, links } })() this.activeDownloads.set(cid.toString(), promise) From 36909ed20326623461aaa6aee784a13ddcc2baac Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 15:52:10 +1200 Subject: [PATCH 14/20] Update type in downloader. --- packages/daemon/src/common/downloader/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/daemon/src/common/downloader/index.ts b/packages/daemon/src/common/downloader/index.ts index af22abf3..ac766f30 100644 --- a/packages/daemon/src/common/downloader/index.ts +++ b/packages/daemon/src/common/downloader/index.ts @@ -59,7 +59,7 @@ export class Downloader implements Startable { } } - private async * batchDownload (itr: AsyncIterable<[CID, number]>): AsyncGenerator<() => Promise<{ cid: CID, block: Uint8Array }>, void, undefined> { + private async * batchDownload (itr: AsyncIterable<[CID, number]>): AsyncGenerator<() => Promise<{ cid: CID }>, void, undefined> { for await (const [cid, priority] of itr) { if (this.isAborted) { return From 8d7299373d0c292f1a01a04cfca279b34a2a124f Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 15:52:49 +1200 Subject: [PATCH 15/20] Fix import path. --- packages/benchmarks/src/transfer/create-transfer-bench.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts index 74540c76..fa746d71 100644 --- a/packages/benchmarks/src/transfer/create-transfer-bench.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -22,7 +22,7 @@ export const createTransferBench = async (size: number): Promise Date: Mon, 8 Apr 2024 15:56:24 +1200 Subject: [PATCH 16/20] Add option for persistent storage to run node. --- .../src/transfer/create-transfer-bench.ts | 2 +- packages/benchmarks/src/utils/run-node.ts | 22 ++++++++++++++++--- 2 files changed, 20 insertions(+), 4 deletions(-) diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/create-transfer-bench.ts index fa746d71..c4af0453 100644 --- a/packages/benchmarks/src/transfer/create-transfer-bench.ts +++ b/packages/benchmarks/src/transfer/create-transfer-bench.ts @@ -13,7 +13,7 @@ export const createTransferBench = async (size: number): Promise p.start())) - const clients = names.map(n => new Client(Path.join(dataPath, n, `socket`))) + const clients = names.map(n => new Client(Path.join(dataPath, n, 'socket'))) const addresses = await clients[0].addresses() diff --git a/packages/benchmarks/src/utils/run-node.ts b/packages/benchmarks/src/utils/run-node.ts index 3f45b115..9bad75e0 100644 --- a/packages/benchmarks/src/utils/run-node.ts +++ b/packages/benchmarks/src/utils/run-node.ts @@ -5,14 +5,30 @@ import { DeferredPromise } from '@open-draft/deferred-promise' import { toString as uint8ArrayToString } from 'uint8arrays' import { packagePath, modulesPath } from './paths.js' -export default async (name: string): Promise<{ start(): Promise, stop(): Promise }> => { - const socket = Path.join(packagePath, `${name}.socket`) +export default async (name: string, options: { persistent?: boolean } = {}): Promise<{ start(): Promise, stop(): Promise }> => { + const rootPath = Path.join(packagePath, 'test-out', name) + const socket = Path.join(rootPath, 'socket') + const config = Path.join(rootPath, 'config.json') + const isReceiver = name[name.length - 1] === '1' + const storagePath = Path.join(rootPath, 'storage') + + await fs.mkdir(rootPath, { recursive: true }) + + await fs.writeFile(config, JSON.stringify({ + storage: options.persistent === true ? storagePath : ':memory:' + })) const args = [ Path.join(modulesPath, '@organicdesign/db-daemon/dist/src/index.js'), - '-s', socket + '-s', socket, + '-c', config ] + if (isReceiver) { + // args.unshift('--trace-gc') + // args.unshift('--inspect') + } + let proc: ChildProcessWithoutNullStreams const forceQuit = async (): Promise => { From a06d026b4a0e74e818dcf84e6b4989c446041e70 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 16:13:47 +1200 Subject: [PATCH 17/20] Rename transfer bench file. --- packages/benchmarks/src/transfer/index.ts | 2 +- .../transfer/{create-transfer-bench.ts => transfer-bench.ts} | 0 2 files changed, 1 insertion(+), 1 deletion(-) rename packages/benchmarks/src/transfer/{create-transfer-bench.ts => transfer-bench.ts} (100%) diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index ed44c9a0..669454d5 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -6,7 +6,7 @@ import prettyBytes from 'pretty-bytes' import { Bench } from 'tinybench' import generateFile from '../utils/generate-file.js' import { packagePath } from '../utils/paths.js' -import { createTransferBench } from './create-transfer-bench.js' +import { createTransferBench } from './transfer-bench.js' import type { TransferImplementation } from './interface.js' const log = debug('bench:transfer') diff --git a/packages/benchmarks/src/transfer/create-transfer-bench.ts b/packages/benchmarks/src/transfer/transfer-bench.ts similarity index 100% rename from packages/benchmarks/src/transfer/create-transfer-bench.ts rename to packages/benchmarks/src/transfer/transfer-bench.ts From 4fd8cb0bc8e72257b9191f394708b4b2a67de740 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 16:25:31 +1200 Subject: [PATCH 18/20] Add import benchmark. --- .../benchmarks/src/import/import-bench.ts | 36 +++++ packages/benchmarks/src/import/index.ts | 128 ++++++++++++++++++ packages/benchmarks/src/import/interface.ts | 17 +++ packages/benchmarks/src/transfer/index.ts | 2 +- 4 files changed, 182 insertions(+), 1 deletion(-) create mode 100644 packages/benchmarks/src/import/import-bench.ts create mode 100644 packages/benchmarks/src/import/index.ts create mode 100644 packages/benchmarks/src/import/interface.ts diff --git a/packages/benchmarks/src/import/import-bench.ts b/packages/benchmarks/src/import/import-bench.ts new file mode 100644 index 00000000..3aa7bf1c --- /dev/null +++ b/packages/benchmarks/src/import/import-bench.ts @@ -0,0 +1,36 @@ +import Path from 'path' +import { Client } from '@organicdesign/db-client' +import { packagePath } from '../utils/paths.js' +import runNode from '../utils/run-node.js' +import type { ImportBenchmark } from './interface.js' + +export const createImportBench = async (size: number): Promise => { + const dataPath = Path.join(packagePath, 'test-out') + + const name = `import-${size}` + + const proc = await runNode(name) + + await proc.start() + + const client = new Client(Path.join(dataPath, name, 'socket')) + + const group = await client.createGroup('test') + + const dataFile = Path.join(dataPath, `${size}.data`) + + return { + async teardown () { + client.stop() + + await proc.stop() + }, + + async run () { + const [{ cid }] = await client.import(group, dataFile, { path: '/test' }) + const [item] = await client.getStatus([cid]) + + return item + } + } +} diff --git a/packages/benchmarks/src/import/index.ts b/packages/benchmarks/src/import/index.ts new file mode 100644 index 00000000..dfcb5487 --- /dev/null +++ b/packages/benchmarks/src/import/index.ts @@ -0,0 +1,128 @@ +/* eslint-disable no-console,no-loop-func */ +import fs from 'fs/promises' +import Path from 'path' +import debug from 'debug' +import prettyBytes from 'pretty-bytes' +import { Bench } from 'tinybench' +import generateFile from '../utils/generate-file.js' +import { packagePath } from '../utils/paths.js' +import { createImportBench } from './import-bench.js' +import type { ImportImplementation } from './interface.js' + +const log = debug('bench:transfer') + +const ITERATIONS = parseInt(process.env.ITERATIONS ?? '3') +const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') +const RESULT_PRECISION = 2 + +const sizes = [ + 0, // 1b + 3, // 1kb + 6, // 1mb + 7, // 10mb + 8, // 100mb + 9 // 1gb +].map(z => 10 ** z) + +const impls: ImportImplementation[] = sizes.map(size => ({ + name: `${prettyBytes(size)}`, + create: async () => createImportBench(size), + results: [], + fileSize: size, + size: 0, + blocks: 0 +})) + +const dataPath = Path.join(packagePath, 'test-out') + +async function main (): Promise { + const suite = new Bench({ + iterations: ITERATIONS, + time: MIN_TIME, + + async setup (task) { + const impl = impls.find(({ name }) => task.name.includes(name)) + + if (impl == null) { + return + } + + await fs.mkdir(dataPath, { recursive: true }) + const dataFile = Path.join(dataPath, `${impl.fileSize}.data`) + await generateFile(dataFile, impl.fileSize) + }, + + async teardown (task) { + const impl = impls.find(({ name }) => task.name.includes(name)) + + if (impl == null) { + return + } + + const dataFile = Path.join(dataPath, `${impl.fileSize}.data`) + await fs.rm(dataFile) + } + }) + + for (const impl of impls) { + const run = async function (): Promise { + log('Start: setup') + const subject = await impl.create() + log('End: setup') + + const start = performance.now() + const { size, blocks } = await subject.run() + + impl.size = size + impl.blocks = blocks + impl.results.push(performance.now() - start) + + log('Start: teardown') + await subject.teardown() + log('End: teardown') + } + + const hooks = { + beforeEach: async () => { + log(`Start: test ${impl.name}`) + }, + afterEach: async () => { + log(`End: test ${impl.name}`) + } + } + + suite.add(impl.name, run, hooks) + } + + await suite.run() + + console.table(suite.tasks.map(({ name, result }) => { + const impl = impls.find(impl => impl.name === name) + + if (impl == null) { + throw new Error('got result without implementation') + } + + const seconds = (result?.period ?? 0) / 1000 + const speed = impl.size / seconds + const bps = impl.blocks / seconds + + return { + 'File Size': name, + Size: prettyBytes(impl.size), + Blocks: impl.blocks, + 'Speed (Size)': `${prettyBytes(speed)}/s`, + 'Speed (Blocks)': `${bps.toFixed(RESULT_PRECISION)} blocks/s`, + 'Run Time': `${result?.period.toFixed(RESULT_PRECISION)}ms`, + Runs: result?.samples.length, + p99: `${result?.p99.toFixed(RESULT_PRECISION)}ms` + } + })) + + await fs.rm(dataPath, { recursive: true }) +} + +main().catch(err => { + console.error(err) // eslint-disable-line no-console + process.exit(1) +}) diff --git a/packages/benchmarks/src/import/interface.ts b/packages/benchmarks/src/import/interface.ts new file mode 100644 index 00000000..ecebf3de --- /dev/null +++ b/packages/benchmarks/src/import/interface.ts @@ -0,0 +1,17 @@ +export interface ImportBenchmark { + teardown(): Promise + run(): Promise<{ + cid: string + blocks: number + size: number + }> +} + +export interface ImportImplementation { + name: string + create(): Promise + results: number[] + fileSize: number + size: number + blocks: number +} diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index 669454d5..04888dc0 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -120,7 +120,7 @@ async function main (): Promise { } })) - await fs.rm(dataPath, { recursive: true }) + await fs.rm(dataPath, { recursive: true }) } main().catch(err => { From e0a6613c25e83abd8646018a2a33d1de65b46354 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 16:42:14 +1200 Subject: [PATCH 19/20] Fix dependencies. --- package-lock.json | 77 +++++++++++++---------- packages/benchmarks/package.json | 4 +- packages/daemon/package.json | 9 ++- packages/helia-pin-manager/package.json | 2 +- packages/key-manager/package.json | 2 +- packages/manual-block-broker/package.json | 10 +-- packages/test-utils/package.json | 7 ++- packages/utils/package.json | 10 +-- 8 files changed, 70 insertions(+), 51 deletions(-) diff --git a/package-lock.json b/package-lock.json index e0a0ecd6..dd16034a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3233,11 +3233,11 @@ } }, "node_modules/@libp2p/interface": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/@libp2p/interface/-/interface-1.1.5.tgz", - "integrity": "sha512-BjFgv/3VwEDNRcFKL4KW6g29IcUWUjaTJhyZVGWtodFuPjZsZHJgoQU7T/FFxDcfTdI90qpFbTREycOB+VL9NQ==", + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/@libp2p/interface/-/interface-1.1.6.tgz", + "integrity": "sha512-CLz6TAZf+Mw1PCIU8pjMIct1uh3A1fIene2/t+E57Tw4uJLCBJE9CLed/Opxliy5RH0e32Aa6bi4QSXtkJTK7A==", "dependencies": { - "@multiformats/multiaddr": "^12.1.14", + "@multiformats/multiaddr": "^12.2.1", "it-pushable": "^3.2.3", "it-stream-types": "^2.0.1", "multiformats": "^13.1.0", @@ -3257,7 +3257,8 @@ }, "node_modules/@libp2p/interfaces": { "version": "3.3.2", - "license": "Apache-2.0 OR MIT", + "resolved": "https://registry.npmjs.org/@libp2p/interfaces/-/interfaces-3.3.2.tgz", + "integrity": "sha512-p/M7plbrxLzuQchvNwww1Was7ZeGE2NaOFulMaZBYIihU8z3fhaV+a033OqnC/0NTX/yhfdNOG7znhYq3XoR/g==", "engines": { "node": ">=16.0.0", "npm": ">=7.0.0" @@ -3653,13 +3654,14 @@ } }, "node_modules/@multiformats/multiaddr": { - "version": "12.1.14", - "license": "Apache-2.0 OR MIT", + "version": "12.2.1", + "resolved": "https://registry.npmjs.org/@multiformats/multiaddr/-/multiaddr-12.2.1.tgz", + "integrity": "sha512-UwjoArBbv64FlaetV4DDwh+PUMfzXUBltxQwdh+uTYnGFzVa8ZfJsn1vt1RJlJ6+Xtrm3RMekF/B+K338i2L5Q==", "dependencies": { "@chainsafe/is-ip": "^2.0.1", "@chainsafe/netmask": "^2.0.0", "@libp2p/interface": "^1.0.0", - "dns-over-http-resolver": "^3.0.2", + "@multiformats/dns": "^1.0.3", "multiformats": "^13.0.0", "uint8-varint": "^2.0.1", "uint8arrays": "^5.0.0" @@ -3909,7 +3911,8 @@ }, "node_modules/@open-draft/deferred-promise": { "version": "2.2.0", - "license": "MIT" + "resolved": "https://registry.npmjs.org/@open-draft/deferred-promise/-/deferred-promise-2.2.0.tgz", + "integrity": "sha512-CecwLWx3rhxVQF6V4bAgPS5t+So2sTbPgAzafKkVizyi7tlwpcFpdFqq+wqF2OwNBmqFuu6tOyouTuxgpMfzmA==" }, "node_modules/@organicdesign/db-benchamrks": { "resolved": "packages/benchmarks", @@ -14040,7 +14043,8 @@ }, "node_modules/interface-blockstore": { "version": "5.2.10", - "license": "Apache-2.0 OR MIT", + "resolved": "https://registry.npmjs.org/interface-blockstore/-/interface-blockstore-5.2.10.tgz", + "integrity": "sha512-9K48hTvBCGsKVD3pF4ILgDcf+W2P/gq0oxLcsHGB6E6W6nDutYkzR+7k7bCs9REHrBEfKzcVDEKieiuNM9WRZg==", "dependencies": { "interface-store": "^5.0.0", "multiformats": "^13.0.1" @@ -14068,7 +14072,8 @@ }, "node_modules/interface-store": { "version": "5.1.8", - "license": "Apache-2.0 OR MIT" + "resolved": "https://registry.npmjs.org/interface-store/-/interface-store-5.1.8.tgz", + "integrity": "sha512-7na81Uxkl0vqk0CBPO5PvyTkdaJBaezwUJGsMOz7riPOq0rJt+7W31iaopaMICWea/iykUsvNlPx/Tc+MxC3/w==" }, "node_modules/internal-slot": { "version": "1.0.7", @@ -15786,7 +15791,8 @@ }, "node_modules/it-parallel": { "version": "3.0.6", - "license": "Apache-2.0 OR MIT", + "resolved": "https://registry.npmjs.org/it-parallel/-/it-parallel-3.0.6.tgz", + "integrity": "sha512-i7UM7I9LTkDJw3YIqXHFAPZX6CWYzGc+X3irdNrVExI4vPazrJdI7t5OqrSVN8CONXLAunCiqaSV/zZRbQR56A==", "dependencies": { "p-defer": "^4.0.0" } @@ -15804,7 +15810,8 @@ }, "node_modules/it-pipe": { "version": "3.0.1", - "license": "Apache-2.0 OR MIT", + "resolved": "https://registry.npmjs.org/it-pipe/-/it-pipe-3.0.1.tgz", + "integrity": "sha512-sIoNrQl1qSRg2seYSBH/3QxWhJFn9PKYvOf/bHdtCBF0bnghey44VyASsWzn5dAx0DCDDABq1hZIuzKmtBZmKA==", "dependencies": { "it-merge": "^3.0.0", "it-pushable": "^3.1.2", @@ -30617,10 +30624,12 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { + "@open-draft/deferred-promise": "^2.2.0", "@organicdesign/db-client": "^0.1.0", "debug": "^4.3.4", "pretty-bytes": "^6.1.1", - "tinybench": "^2.6.0" + "tinybench": "^2.6.0", + "uint8arrays": "^5.0.3" }, "devDependencies": { "aegir": "^42.2.4" @@ -30677,6 +30686,7 @@ "@chainsafe/libp2p-yamux": "^6.0.2", "@helia/block-brokers": "^2.0.3", "@helia/car": "^3.1.0", + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.0", "@ipld/car": "^5.3.0", "@ipld/dag-cbor": "^9.2.0", @@ -30685,6 +30695,8 @@ "@libp2p/circuit-relay-v2": "^1.0.15", "@libp2p/dcutr": "^1.0.12", "@libp2p/identify": "^1.0.14", + "@libp2p/interface": "^1.1.6", + "@libp2p/interfaces": "^3.3.2", "@libp2p/kad-dht": "^12.0.7", "@libp2p/logger": "^4.0.6", "@libp2p/pnet": "^1.0.1", @@ -30705,6 +30717,7 @@ "datastore-fs": "^9.1.8", "debug": "^4.3.4", "helia": "^4.0.1", + "interface-blockstore": "^5.2.10", "interface-datastore": "^8.2.11", "it-all": "^3.0.4", "it-parallel": "^3.0.6", @@ -30720,14 +30733,9 @@ "zod": "^3.22.4" }, "devDependencies": { - "@helia/interface": "^4.0.0", - "@ipld/dag-pb": "^4.1.0", - "@libp2p/interface": "^1.1.3", - "@libp2p/interfaces": "^3.3.2", "@organicdesign/db-test-utils": "^0.1.0", "@types/yargs": "^17.0.32", "aegir": "^42.2.4", - "interface-blockstore": "^5.2.10", "tsc-alias": "^1.8.8" } }, @@ -30859,6 +30867,7 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { + "@helia/interface": "^4.1.0", "@open-draft/deferred-promise": "^2.2.0", "@organicdesign/db-utils": "^0.1.0", "cborg": "^4.0.9", @@ -30871,7 +30880,6 @@ "zod": "^3.22.4" }, "devDependencies": { - "@helia/interface": "^4.0.0", "@ipld/dag-pb": "^4.1.0", "aegir": "^42.2.4", "helia": "^4.0.1", @@ -30886,6 +30894,7 @@ "dependencies": { "@ipld/dag-cbor": "^9.2.0", "@libp2p/crypto": "^4.0.2", + "@libp2p/interface": "^1.1.6", "@libp2p/peer-id": "^4.0.6", "bip32": "^4.0.0", "bip39": "^3.1.0", @@ -30896,7 +30905,6 @@ "zod": "^3.22.4" }, "devDependencies": { - "@libp2p/interface": "^1.1.3", "aegir": "^42.2.4" } }, @@ -30905,13 +30913,13 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { - "@open-draft/deferred-promise": "^2.2.0" - }, - "devDependencies": { - "@helia/interface": "^4.0.0", - "aegir": "^42.2.4", + "@helia/interface": "^4.1.0", + "@open-draft/deferred-promise": "^2.2.0", "interface-store": "^5.1.8", "multiformats": "^13.1.0" + }, + "devDependencies": { + "aegir": "^42.2.4" } }, "packages/rpc-interfaces": { @@ -30962,14 +30970,17 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.2", + "@ipld/dag-pb": "^4.1.0", "blockstore-core": "^4.4.0", + "interface-blockstore": "^5.2.10", "it-all": "^3.0.4", + "multiformats": "^13.1.0", "uint8arrays": "^5.0.3" }, "devDependencies": { - "aegir": "^42.2.4", - "multiformats": "^13.1.0" + "aegir": "^42.2.4" } }, "packages/utils": { @@ -30977,25 +30988,27 @@ "version": "0.1.0", "license": "GPL-3.0-or-later", "dependencies": { + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.2", "@ipld/dag-cbor": "^9.2.0", "@ipld/dag-json": "^10.2.0", "@ipld/dag-pb": "^4.1.0", + "@libp2p/interface": "^1.1.6", "cborg": "^4.1.4", "datastore-core": "^9.2.9", + "interface-blockstore": "^5.2.10", "interface-datastore": "^8.2.11", "ipfs-unixfs-exporter": "^13.5.0", "ipfs-unixfs-importer": "^15.2.4", + "it-all": "^3.0.4", "multiformats": "^13.1.0" }, "devDependencies": { - "@helia/interface": "^4.1.0", - "@libp2p/interface": "^1.1.5", "@organicdesign/db-test-utils": "^0.1.0", "aegir": "^42.2.4", "blockstore-core": "^4.4.0", - "interface-blockstore": "^5.2.10", - "it-all": "^3.0.4" + "it-parallel": "^3.0.6", + "it-pipe": "^3.0.1" } } } diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json index 3583f5c3..bc8ddd9c 100644 --- a/packages/benchmarks/package.json +++ b/packages/benchmarks/package.json @@ -29,10 +29,12 @@ "private": true, "sideEffects": false, "dependencies": { + "@open-draft/deferred-promise": "^2.2.0", "@organicdesign/db-client": "^0.1.0", "debug": "^4.3.4", "pretty-bytes": "^6.1.1", - "tinybench": "^2.6.0" + "tinybench": "^2.6.0", + "uint8arrays": "^5.0.3" }, "peerDependencies": { "@organicdesign/db-daemon": "^0.1.0" diff --git a/packages/daemon/package.json b/packages/daemon/package.json index 5bf03e1d..07d32474 100644 --- a/packages/daemon/package.json +++ b/packages/daemon/package.json @@ -25,14 +25,9 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "@helia/interface": "^4.0.0", - "@ipld/dag-pb": "^4.1.0", - "@libp2p/interface": "^1.1.3", - "@libp2p/interfaces": "^3.3.2", "@organicdesign/db-test-utils": "^0.1.0", "@types/yargs": "^17.0.32", "aegir": "^42.2.4", - "interface-blockstore": "^5.2.10", "tsc-alias": "^1.8.8" }, "private": true, @@ -43,6 +38,7 @@ "@chainsafe/libp2p-yamux": "^6.0.2", "@helia/block-brokers": "^2.0.3", "@helia/car": "^3.1.0", + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.0", "@ipld/car": "^5.3.0", "@ipld/dag-cbor": "^9.2.0", @@ -51,6 +47,8 @@ "@libp2p/circuit-relay-v2": "^1.0.15", "@libp2p/dcutr": "^1.0.12", "@libp2p/identify": "^1.0.14", + "@libp2p/interface": "^1.1.6", + "@libp2p/interfaces": "^3.3.2", "@libp2p/kad-dht": "^12.0.7", "@libp2p/logger": "^4.0.6", "@libp2p/pnet": "^1.0.1", @@ -71,6 +69,7 @@ "datastore-fs": "^9.1.8", "debug": "^4.3.4", "helia": "^4.0.1", + "interface-blockstore": "^5.2.10", "interface-datastore": "^8.2.11", "it-all": "^3.0.4", "it-parallel": "^3.0.6", diff --git a/packages/helia-pin-manager/package.json b/packages/helia-pin-manager/package.json index d9839f8d..2fcf1322 100644 --- a/packages/helia-pin-manager/package.json +++ b/packages/helia-pin-manager/package.json @@ -25,7 +25,6 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "@helia/interface": "^4.0.0", "@ipld/dag-pb": "^4.1.0", "aegir": "^42.2.4", "helia": "^4.0.1", @@ -35,6 +34,7 @@ "private": true, "sideEffects": false, "dependencies": { + "@helia/interface": "^4.1.0", "@open-draft/deferred-promise": "^2.2.0", "@organicdesign/db-utils": "^0.1.0", "cborg": "^4.0.9", diff --git a/packages/key-manager/package.json b/packages/key-manager/package.json index bce300ea..a074bee8 100644 --- a/packages/key-manager/package.json +++ b/packages/key-manager/package.json @@ -25,7 +25,6 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "@libp2p/interface": "^1.1.3", "aegir": "^42.2.4" }, "private": true, @@ -33,6 +32,7 @@ "dependencies": { "@ipld/dag-cbor": "^9.2.0", "@libp2p/crypto": "^4.0.2", + "@libp2p/interface": "^1.1.6", "@libp2p/peer-id": "^4.0.6", "bip32": "^4.0.0", "bip39": "^3.1.0", diff --git a/packages/manual-block-broker/package.json b/packages/manual-block-broker/package.json index 307ba0c7..b9b4006e 100644 --- a/packages/manual-block-broker/package.json +++ b/packages/manual-block-broker/package.json @@ -25,14 +25,14 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "@helia/interface": "^4.0.0", - "aegir": "^42.2.4", - "interface-store": "^5.1.8", - "multiformats": "^13.1.0" + "aegir": "^42.2.4" }, "private": true, "sideEffects": false, "dependencies": { - "@open-draft/deferred-promise": "^2.2.0" + "@helia/interface": "^4.1.0", + "@open-draft/deferred-promise": "^2.2.0", + "interface-store": "^5.1.8", + "multiformats": "^13.1.0" } } diff --git a/packages/test-utils/package.json b/packages/test-utils/package.json index 3afa4227..2fafc196 100644 --- a/packages/test-utils/package.json +++ b/packages/test-utils/package.json @@ -28,15 +28,18 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "aegir": "^42.2.4", - "multiformats": "^13.1.0" + "aegir": "^42.2.4" }, "private": true, "sideEffects": false, "dependencies": { + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.2", + "@ipld/dag-pb": "^4.1.0", "blockstore-core": "^4.4.0", + "interface-blockstore": "^5.2.10", "it-all": "^3.0.4", + "multiformats": "^13.1.0", "uint8arrays": "^5.0.3" } } diff --git a/packages/utils/package.json b/packages/utils/package.json index dc38cbe9..aed4b3d3 100644 --- a/packages/utils/package.json +++ b/packages/utils/package.json @@ -33,26 +33,28 @@ "author": "Saul Boyd", "license": "GPL-3.0-or-later", "devDependencies": { - "@helia/interface": "^4.1.0", - "@libp2p/interface": "^1.1.5", "@organicdesign/db-test-utils": "^0.1.0", "aegir": "^42.2.4", "blockstore-core": "^4.4.0", - "interface-blockstore": "^5.2.10", - "it-all": "^3.0.4" + "it-parallel": "^3.0.6", + "it-pipe": "^3.0.1" }, "private": true, "sideEffects": false, "dependencies": { + "@helia/interface": "^4.1.0", "@helia/unixfs": "^3.0.2", "@ipld/dag-cbor": "^9.2.0", "@ipld/dag-json": "^10.2.0", "@ipld/dag-pb": "^4.1.0", + "@libp2p/interface": "^1.1.6", "cborg": "^4.1.4", "datastore-core": "^9.2.9", + "interface-blockstore": "^5.2.10", "interface-datastore": "^8.2.11", "ipfs-unixfs-exporter": "^13.5.0", "ipfs-unixfs-importer": "^15.2.4", + "it-all": "^3.0.4", "multiformats": "^13.1.0" } } From 44e220c3ea5828bbd2a7aa0dd22a1f98b9acd587 Mon Sep 17 00:00:00 2001 From: saul Date: Mon, 8 Apr 2024 16:55:40 +1200 Subject: [PATCH 20/20] Replace environment vars with arguments. --- package-lock.json | 6 ++- packages/benchmarks/package.json | 3 +- .../benchmarks/src/import/import-bench.ts | 4 +- packages/benchmarks/src/import/index.ts | 47 +++++++++++++++---- packages/benchmarks/src/transfer/index.ts | 47 +++++++++++++++---- .../benchmarks/src/transfer/transfer-bench.ts | 4 +- 6 files changed, 84 insertions(+), 27 deletions(-) diff --git a/package-lock.json b/package-lock.json index dd16034a..9e456253 100644 --- a/package-lock.json +++ b/package-lock.json @@ -30503,7 +30503,8 @@ }, "node_modules/yargs": { "version": "17.7.2", - "license": "MIT", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", @@ -30629,7 +30630,8 @@ "debug": "^4.3.4", "pretty-bytes": "^6.1.1", "tinybench": "^2.6.0", - "uint8arrays": "^5.0.3" + "uint8arrays": "^5.0.3", + "yargs": "^17.7.2" }, "devDependencies": { "aegir": "^42.2.4" diff --git a/packages/benchmarks/package.json b/packages/benchmarks/package.json index bc8ddd9c..1eb786cd 100644 --- a/packages/benchmarks/package.json +++ b/packages/benchmarks/package.json @@ -34,7 +34,8 @@ "debug": "^4.3.4", "pretty-bytes": "^6.1.1", "tinybench": "^2.6.0", - "uint8arrays": "^5.0.3" + "uint8arrays": "^5.0.3", + "yargs": "^17.7.2" }, "peerDependencies": { "@organicdesign/db-daemon": "^0.1.0" diff --git a/packages/benchmarks/src/import/import-bench.ts b/packages/benchmarks/src/import/import-bench.ts index 3aa7bf1c..41d2895f 100644 --- a/packages/benchmarks/src/import/import-bench.ts +++ b/packages/benchmarks/src/import/import-bench.ts @@ -4,12 +4,12 @@ import { packagePath } from '../utils/paths.js' import runNode from '../utils/run-node.js' import type { ImportBenchmark } from './interface.js' -export const createImportBench = async (size: number): Promise => { +export const createImportBench = async (size: number, persistent: boolean): Promise => { const dataPath = Path.join(packagePath, 'test-out') const name = `import-${size}` - const proc = await runNode(name) + const proc = await runNode(name, { persistent }) await proc.start() diff --git a/packages/benchmarks/src/import/index.ts b/packages/benchmarks/src/import/index.ts index dfcb5487..dc716073 100644 --- a/packages/benchmarks/src/import/index.ts +++ b/packages/benchmarks/src/import/index.ts @@ -4,16 +4,43 @@ import Path from 'path' import debug from 'debug' import prettyBytes from 'pretty-bytes' import { Bench } from 'tinybench' +import { hideBin } from 'yargs/helpers' +import yargs from 'yargs/yargs' import generateFile from '../utils/generate-file.js' import { packagePath } from '../utils/paths.js' import { createImportBench } from './import-bench.js' import type { ImportImplementation } from './interface.js' -const log = debug('bench:transfer') +const argv = await yargs(hideBin(process.argv)) + .option({ + iterations: { + alias: 'i', + type: 'number', + default: 3 + } + }) + .option({ + minTime: { + type: 'number', + default: 1 + } + }) + .option({ + precision: { + type: 'number', + default: 2 + } + }) + .option({ + persistent: { + alias: 'p', + type: 'boolean', + default: false + } + }) + .parse() -const ITERATIONS = parseInt(process.env.ITERATIONS ?? '3') -const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') -const RESULT_PRECISION = 2 +const log = debug('bench:import') const sizes = [ 0, // 1b @@ -26,7 +53,7 @@ const sizes = [ const impls: ImportImplementation[] = sizes.map(size => ({ name: `${prettyBytes(size)}`, - create: async () => createImportBench(size), + create: async () => createImportBench(size, argv.persistent), results: [], fileSize: size, size: 0, @@ -37,8 +64,8 @@ const dataPath = Path.join(packagePath, 'test-out') async function main (): Promise { const suite = new Bench({ - iterations: ITERATIONS, - time: MIN_TIME, + iterations: argv.iterations, + time: argv.minTime, async setup (task) { const impl = impls.find(({ name }) => task.name.includes(name)) @@ -112,10 +139,10 @@ async function main (): Promise { Size: prettyBytes(impl.size), Blocks: impl.blocks, 'Speed (Size)': `${prettyBytes(speed)}/s`, - 'Speed (Blocks)': `${bps.toFixed(RESULT_PRECISION)} blocks/s`, - 'Run Time': `${result?.period.toFixed(RESULT_PRECISION)}ms`, + 'Speed (Blocks)': `${bps.toFixed(argv.precision)} blocks/s`, + 'Run Time': `${result?.period.toFixed(argv.precision)}ms`, Runs: result?.samples.length, - p99: `${result?.p99.toFixed(RESULT_PRECISION)}ms` + p99: `${result?.p99.toFixed(argv.precision)}ms` } })) diff --git a/packages/benchmarks/src/transfer/index.ts b/packages/benchmarks/src/transfer/index.ts index 04888dc0..a0cf738e 100644 --- a/packages/benchmarks/src/transfer/index.ts +++ b/packages/benchmarks/src/transfer/index.ts @@ -4,16 +4,43 @@ import Path from 'path' import debug from 'debug' import prettyBytes from 'pretty-bytes' import { Bench } from 'tinybench' +import { hideBin } from 'yargs/helpers' +import yargs from 'yargs/yargs' import generateFile from '../utils/generate-file.js' import { packagePath } from '../utils/paths.js' import { createTransferBench } from './transfer-bench.js' import type { TransferImplementation } from './interface.js' -const log = debug('bench:transfer') +const argv = await yargs(hideBin(process.argv)) + .option({ + iterations: { + alias: 'i', + type: 'number', + default: 3 + } + }) + .option({ + minTime: { + type: 'number', + default: 1 + } + }) + .option({ + precision: { + type: 'number', + default: 2 + } + }) + .option({ + persistent: { + alias: 'p', + type: 'boolean', + default: false + } + }) + .parse() -const ITERATIONS = parseInt(process.env.ITERATIONS ?? '3') -const MIN_TIME = parseInt(process.env.MIN_TIME ?? '1') -const RESULT_PRECISION = 2 +const log = debug('bench:transfer') const sizes = [ 0, // 1b @@ -26,7 +53,7 @@ const sizes = [ const impls: TransferImplementation[] = sizes.map(size => ({ name: `${prettyBytes(size)}`, - create: async () => createTransferBench(size), + create: async () => createTransferBench(size, argv.persistent), results: [], fileSize: size, size: 0, @@ -37,8 +64,8 @@ const dataPath = Path.join(packagePath, 'test-out') async function main (): Promise { const suite = new Bench({ - iterations: ITERATIONS, - time: MIN_TIME, + iterations: argv.iterations, + time: argv.minTime, async setup (task) { const impl = impls.find(({ name }) => task.name.includes(name)) @@ -113,10 +140,10 @@ async function main (): Promise { Size: prettyBytes(impl.size), Blocks: impl.blocks, 'Speed (Size)': `${prettyBytes(speed)}/s`, - 'Speed (Blocks)': `${bps.toFixed(RESULT_PRECISION)} blocks/s`, - 'Run Time': `${result?.period.toFixed(RESULT_PRECISION)}ms`, + 'Speed (Blocks)': `${bps.toFixed(argv.precision)} blocks/s`, + 'Run Time': `${result?.period.toFixed(argv.precision)}ms`, Runs: result?.samples.length, - p99: `${result?.p99.toFixed(RESULT_PRECISION)}ms` + p99: `${result?.p99.toFixed(argv.precision)}ms` } })) diff --git a/packages/benchmarks/src/transfer/transfer-bench.ts b/packages/benchmarks/src/transfer/transfer-bench.ts index c4af0453..b7f47ee3 100644 --- a/packages/benchmarks/src/transfer/transfer-bench.ts +++ b/packages/benchmarks/src/transfer/transfer-bench.ts @@ -4,12 +4,12 @@ import { packagePath } from '../utils/paths.js' import runNode from '../utils/run-node.js' import type { TransferBenchmark } from './interface.js' -export const createTransferBench = async (size: number): Promise => { +export const createTransferBench = async (size: number, persistent: boolean): Promise => { const dataPath = Path.join(packagePath, 'test-out') const names = [...Array(2).keys()].map(i => `transfer-${size}-${i}`) - const procs = await Promise.all(names.map(async n => runNode(n))) + const procs = await Promise.all(names.map(async n => runNode(n, { persistent }))) await Promise.all(procs.map(async p => p.start()))