@hologit/holo-tree 0.0.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 (4) hide show
  1. package/README.md +125 -0
  2. package/index.d.ts +77 -0
  3. package/index.js +317 -0
  4. package/package.json +47 -0
package/README.md ADDED
@@ -0,0 +1,125 @@
1
+ # holo-tree-napi
2
+
3
+ Node.js native binding for [`holo-tree`](../holo-tree) — mutable in-memory git
4
+ trees via [gitoxide](https://github.com/GitoxideLabs/gitoxide), with no `git`
5
+ subprocess.
6
+
7
+ This crate exposes the narrow slice of holo-tree that record-oriented consumers
8
+ (notably [gitsheets](https://github.com/JarvusInnovations/gitsheets)) need for an
9
+ upsert→commit path. It is intentionally a thin pass-through; it is also the first
10
+ integrated consumer of the new Rust libs, so it doubles as a hardening vehicle —
11
+ rough edges in holo-tree's API are recorded as `Phase-C finding` notes in the
12
+ source and fixed upstream rather than worked around here.
13
+
14
+ ## API
15
+
16
+ ```js
17
+ const { Repo, emptyTreeHash } = require('@hologit/holo-tree');
18
+
19
+ const repo = Repo.open('/path/to/repo/.git');
20
+
21
+ const tree = repo.createTreeFromRef('HEAD'); // or repo.createTree() for empty
22
+ tree.writeChild('data/widgets/1.toml', 'id = 1\n'); // hash blob + deep insert
23
+ const treeHash = tree.write(); // flush dirty subtrees → ODB, returns tree hash
24
+
25
+ const commit = repo.commitTree(treeHash, [parentHash], 'add widget 1');
26
+ repo.updateRef('refs/heads/main', commit);
27
+
28
+ const bytes = repo.createTreeFromRef(commit).readBlob('data/widgets/1.toml');
29
+ ```
30
+
31
+ Conventions: object ids cross the boundary as lowercase 40-char hex strings;
32
+ blob content crosses as `Buffer` (binary-safe).
33
+
34
+ ### Surface
35
+
36
+ | Method | holo-tree call | Notes |
37
+ |---|---|---|
38
+ | `Repo.open(gitDir)` | `gix::open().into_sync()` | factory |
39
+ | `repo.createTreeFromRef(ref)` → `Tree` | `repo::create_tree_from_ref` | resolves ref→commit→tree |
40
+ | `repo.createTree()` → `Tree` | `MutableTree::empty` | |
41
+ | `repo.commitTree(treeHash, parents[], msg)` → hash | `repo::commit_tree` | uses git-config identity |
42
+ | `repo.updateRef(ref, hash)` | `repo::update_ref` | |
43
+ | `tree.writeChild(path, text)` → hash | `MutableTree::write_child` | UTF-8 text |
44
+ | `tree.writeChildBytes(path, buf)` → hash | `MutableTree::write_child_bytes` | binary |
45
+ | `tree.readBlob(path)` → `Buffer\|null` | `MutableTree::read_blob` | |
46
+ | `tree.deleteChildDeep(path)` → bool | `MutableTree::delete_child_deep` | |
47
+ | `tree.write()` → treeHash | `MutableTree::write` | |
48
+ | `emptyTreeHash()` → hash | `tree::empty_tree_id` | module fn |
49
+
50
+ ## Building
51
+
52
+ Requires a Rust toolchain and `@napi-rs/cli` (a devDependency):
53
+
54
+ ```sh
55
+ npm install
56
+ npm run build:debug # or: npm run build (release)
57
+ npm test # node --test against a scratch git repo
58
+ ```
59
+
60
+ `napi build` emits `holo-tree.<triple>.node`. The generated `index.js` loader
61
+ and `index.d.ts` types **are committed**; only the `.node` binaries are
62
+ git-ignored (built per-platform in CI).
63
+
64
+ ## Publishing
65
+
66
+ Published as the scoped package **`@hologit/holo-tree`** with per-platform
67
+ prebuilt binaries shipped as `optionalDependencies`:
68
+
69
+ | Platform package | Triple | Built on |
70
+ | --- | --- | --- |
71
+ | `@hologit/holo-tree-linux-x64-gnu` | `x86_64-unknown-linux-gnu` | ubuntu-latest |
72
+ | `@hologit/holo-tree-darwin-arm64` | `aarch64-apple-darwin` | macos-latest |
73
+ | `@hologit/holo-tree-win32-x64-msvc` | `x86_64-pc-windows-msvc` | windows-latest |
74
+
75
+ Each target builds **natively** on its runner — no cross-compilation. The
76
+ `.github/workflows/holo-tree-napi.yml` workflow builds + smoke-tests all three on
77
+ every PR touching the binding, and on a `holo-tree-v*` tag it builds then
78
+ publishes.
79
+
80
+ Auth is **npm trusted publishing (OIDC)** — no tokens, matching hologit's
81
+ `publish-npm.yml`. Trusted publishing is configured *per package*, and a package
82
+ can't get a trusted publisher until it exists — so the four packages need a
83
+ **one-time manual bootstrap** before automated releases work.
84
+
85
+ ### One-time bootstrap (manual first publish, then configure trusted publishing)
86
+
87
+ The four packages all start at an early version (currently `0.0.1`). They must
88
+ exist on npm before trusted publishing can be turned on.
89
+
90
+ 1. **Get the prebuilt binaries.** Run the `holo-tree-napi` workflow (push the
91
+ branch / open a PR, or trigger `workflow_dispatch`) and download its three
92
+ `bindings-*` artifacts — they hold the `.node` for each platform. A single
93
+ machine can't build all three natively, so use the CI artifacts.
94
+
95
+ 2. **Publish all four manually**, logged in as an `@hologit` org member
96
+ (`npm login`):
97
+
98
+ ```sh
99
+ cd holo-tree-napi
100
+ npm install
101
+ npx napi artifacts --dir <downloaded-artifacts-dir> # → npm/<triple>/*.node
102
+ # platform packages first, then the main package:
103
+ for d in npm/*/ ; do ( cd "$d" && npm publish --access public ); done
104
+ npm publish --access public --ignore-scripts # main; skip the napi
105
+ # prepublish GH-release hook
106
+ ```
107
+
108
+ 3. **Turn on trusted publishing** on npmjs.com for **each** of the four packages
109
+ → Settings → Trusted Publisher → GitHub Actions, repo
110
+ `JarvusInnovations/hologit`, workflow `holo-tree-napi.yml`.
111
+
112
+ ### Releases (after bootstrap — fully automated, tokenless)
113
+
114
+ ```sh
115
+ git tag holo-tree-v0.1.0 && git push origin holo-tree-v0.1.0
116
+ ```
117
+
118
+ The tag drives the published version; CI builds all three platforms, then
119
+ publishes via OIDC (provenance + a GitHub release). No secret needed.
120
+
121
+ To add or drop a platform later, edit `napi.triples.additional` +
122
+ `optionalDependencies` in `package.json`, run `napi create-npm-dir -t .`, add the
123
+ matching matrix entry in the workflow, and (since it's a new package) bootstrap
124
+
125
+ + trust that one package too.
package/index.d.ts ADDED
@@ -0,0 +1,77 @@
1
+ /* tslint:disable */
2
+ /* eslint-disable */
3
+
4
+ /* auto-generated by NAPI-RS */
5
+
6
+ /** Git's well-known empty-tree hash (`4b825dc6…`). */
7
+ export declare function emptyTreeHash(): string
8
+ /**
9
+ * A commit identity (author or committer). `timeSeconds`/`offsetMinutes` are
10
+ * optional; when omitted the current wall-clock time at UTC is used. Pass them
11
+ * explicitly to reproduce a specific commit (e.g. match `git commit-tree`
12
+ * under pinned `GIT_AUTHOR_DATE`/`GIT_COMMITTER_DATE`).
13
+ */
14
+ export interface Signature {
15
+ name: string
16
+ email: string
17
+ timeSeconds?: number
18
+ offsetMinutes?: number
19
+ }
20
+ /**
21
+ * A handle to a git repository, backed by gix.
22
+ *
23
+ * Stored as a `ThreadSafeRepository` so the handle is `Send + Sync` and can be
24
+ * cheaply cloned into each `Tree`; every call derives a thread-local
25
+ * `gix::Repository` via `to_thread_local()`.
26
+ */
27
+ export declare class Repo {
28
+ /**
29
+ * Open a repository at `gitDir` (a `.git` directory, or any path gix can
30
+ * discover a repo from).
31
+ */
32
+ static open(gitDir: string): Repo
33
+ /**
34
+ * Resolve a ref (branch, tag, or commit hash) to its tree and return a
35
+ * mutable, in-memory view of it.
36
+ */
37
+ createTreeFromRef(gitRef: string): Tree
38
+ /** Create a fresh empty, mutable in-memory tree rooted at this repo. */
39
+ createTree(): Tree
40
+ /**
41
+ * Write a commit object pointing at `treeHash` with `parents`. `author`
42
+ * and `committer` are optional; each falls back to the repo's configured
43
+ * identity, then a "holo-tree" default. Returns the new commit hash.
44
+ */
45
+ commitTree(treeHash: string, parents: Array<string>, message: string, author?: Signature | undefined | null, committer?: Signature | undefined | null): string
46
+ /** Point a ref at an object hash. */
47
+ updateRef(refname: string, hash: string): void
48
+ }
49
+ /**
50
+ * A mutable, in-memory git tree.
51
+ *
52
+ * Holds its own clone of the repo handle so JS callers don't thread a repo
53
+ * argument through every call.
54
+ *
55
+ * Phase-C finding #1: holo-tree's `MutableTree` takes `&gix::Repository` on
56
+ * nearly every method and keeps a *thread-local* tree cache. We smooth the
57
+ * first half here (the handle lives on the `Tree`) but NOT the second: each
58
+ * call does `to_thread_local()`, and whether holo-tree's thread-local cache
59
+ * stays warm across libuv-dispatched calls is the open ergonomics question to
60
+ * resolve upstream (e.g. a repo-bound tree handle, or an explicit session/
61
+ * cache object the consumer owns).
62
+ */
63
+ export declare class Tree {
64
+ /**
65
+ * Hash `content` (UTF-8 text) as a blob and insert it at `path`, creating
66
+ * intermediate trees as needed. Returns the blob hash.
67
+ */
68
+ writeChild(path: string, content: string): string
69
+ /** Hash raw bytes as a blob and insert at `path`. Binary-safe. */
70
+ writeChildBytes(path: string, content: Buffer): string
71
+ /** Read a blob's bytes at `path`, or `null` if no blob exists there. */
72
+ readBlob(path: string): Buffer | null
73
+ /** Delete a child at a deep `path`. Returns whether it existed. */
74
+ deleteChildDeep(path: string): boolean
75
+ /** Flush dirty subtrees to the ODB and return the resulting tree hash. */
76
+ write(): string
77
+ }
package/index.js ADDED
@@ -0,0 +1,317 @@
1
+ /* tslint:disable */
2
+ /* eslint-disable */
3
+ /* prettier-ignore */
4
+
5
+ /* auto-generated by NAPI-RS */
6
+
7
+ const { existsSync, readFileSync } = require('fs')
8
+ const { join } = require('path')
9
+
10
+ const { platform, arch } = process
11
+
12
+ let nativeBinding = null
13
+ let localFileExisted = false
14
+ let loadError = null
15
+
16
+ function isMusl() {
17
+ // For Node 10
18
+ if (!process.report || typeof process.report.getReport !== 'function') {
19
+ try {
20
+ const lddPath = require('child_process').execSync('which ldd').toString().trim()
21
+ return readFileSync(lddPath, 'utf8').includes('musl')
22
+ } catch (e) {
23
+ return true
24
+ }
25
+ } else {
26
+ const { glibcVersionRuntime } = process.report.getReport().header
27
+ return !glibcVersionRuntime
28
+ }
29
+ }
30
+
31
+ switch (platform) {
32
+ case 'android':
33
+ switch (arch) {
34
+ case 'arm64':
35
+ localFileExisted = existsSync(join(__dirname, 'holo-tree.android-arm64.node'))
36
+ try {
37
+ if (localFileExisted) {
38
+ nativeBinding = require('./holo-tree.android-arm64.node')
39
+ } else {
40
+ nativeBinding = require('@hologit/holo-tree-android-arm64')
41
+ }
42
+ } catch (e) {
43
+ loadError = e
44
+ }
45
+ break
46
+ case 'arm':
47
+ localFileExisted = existsSync(join(__dirname, 'holo-tree.android-arm-eabi.node'))
48
+ try {
49
+ if (localFileExisted) {
50
+ nativeBinding = require('./holo-tree.android-arm-eabi.node')
51
+ } else {
52
+ nativeBinding = require('@hologit/holo-tree-android-arm-eabi')
53
+ }
54
+ } catch (e) {
55
+ loadError = e
56
+ }
57
+ break
58
+ default:
59
+ throw new Error(`Unsupported architecture on Android ${arch}`)
60
+ }
61
+ break
62
+ case 'win32':
63
+ switch (arch) {
64
+ case 'x64':
65
+ localFileExisted = existsSync(
66
+ join(__dirname, 'holo-tree.win32-x64-msvc.node')
67
+ )
68
+ try {
69
+ if (localFileExisted) {
70
+ nativeBinding = require('./holo-tree.win32-x64-msvc.node')
71
+ } else {
72
+ nativeBinding = require('@hologit/holo-tree-win32-x64-msvc')
73
+ }
74
+ } catch (e) {
75
+ loadError = e
76
+ }
77
+ break
78
+ case 'ia32':
79
+ localFileExisted = existsSync(
80
+ join(__dirname, 'holo-tree.win32-ia32-msvc.node')
81
+ )
82
+ try {
83
+ if (localFileExisted) {
84
+ nativeBinding = require('./holo-tree.win32-ia32-msvc.node')
85
+ } else {
86
+ nativeBinding = require('@hologit/holo-tree-win32-ia32-msvc')
87
+ }
88
+ } catch (e) {
89
+ loadError = e
90
+ }
91
+ break
92
+ case 'arm64':
93
+ localFileExisted = existsSync(
94
+ join(__dirname, 'holo-tree.win32-arm64-msvc.node')
95
+ )
96
+ try {
97
+ if (localFileExisted) {
98
+ nativeBinding = require('./holo-tree.win32-arm64-msvc.node')
99
+ } else {
100
+ nativeBinding = require('@hologit/holo-tree-win32-arm64-msvc')
101
+ }
102
+ } catch (e) {
103
+ loadError = e
104
+ }
105
+ break
106
+ default:
107
+ throw new Error(`Unsupported architecture on Windows: ${arch}`)
108
+ }
109
+ break
110
+ case 'darwin':
111
+ localFileExisted = existsSync(join(__dirname, 'holo-tree.darwin-universal.node'))
112
+ try {
113
+ if (localFileExisted) {
114
+ nativeBinding = require('./holo-tree.darwin-universal.node')
115
+ } else {
116
+ nativeBinding = require('@hologit/holo-tree-darwin-universal')
117
+ }
118
+ break
119
+ } catch {}
120
+ switch (arch) {
121
+ case 'x64':
122
+ localFileExisted = existsSync(join(__dirname, 'holo-tree.darwin-x64.node'))
123
+ try {
124
+ if (localFileExisted) {
125
+ nativeBinding = require('./holo-tree.darwin-x64.node')
126
+ } else {
127
+ nativeBinding = require('@hologit/holo-tree-darwin-x64')
128
+ }
129
+ } catch (e) {
130
+ loadError = e
131
+ }
132
+ break
133
+ case 'arm64':
134
+ localFileExisted = existsSync(
135
+ join(__dirname, 'holo-tree.darwin-arm64.node')
136
+ )
137
+ try {
138
+ if (localFileExisted) {
139
+ nativeBinding = require('./holo-tree.darwin-arm64.node')
140
+ } else {
141
+ nativeBinding = require('@hologit/holo-tree-darwin-arm64')
142
+ }
143
+ } catch (e) {
144
+ loadError = e
145
+ }
146
+ break
147
+ default:
148
+ throw new Error(`Unsupported architecture on macOS: ${arch}`)
149
+ }
150
+ break
151
+ case 'freebsd':
152
+ if (arch !== 'x64') {
153
+ throw new Error(`Unsupported architecture on FreeBSD: ${arch}`)
154
+ }
155
+ localFileExisted = existsSync(join(__dirname, 'holo-tree.freebsd-x64.node'))
156
+ try {
157
+ if (localFileExisted) {
158
+ nativeBinding = require('./holo-tree.freebsd-x64.node')
159
+ } else {
160
+ nativeBinding = require('@hologit/holo-tree-freebsd-x64')
161
+ }
162
+ } catch (e) {
163
+ loadError = e
164
+ }
165
+ break
166
+ case 'linux':
167
+ switch (arch) {
168
+ case 'x64':
169
+ if (isMusl()) {
170
+ localFileExisted = existsSync(
171
+ join(__dirname, 'holo-tree.linux-x64-musl.node')
172
+ )
173
+ try {
174
+ if (localFileExisted) {
175
+ nativeBinding = require('./holo-tree.linux-x64-musl.node')
176
+ } else {
177
+ nativeBinding = require('@hologit/holo-tree-linux-x64-musl')
178
+ }
179
+ } catch (e) {
180
+ loadError = e
181
+ }
182
+ } else {
183
+ localFileExisted = existsSync(
184
+ join(__dirname, 'holo-tree.linux-x64-gnu.node')
185
+ )
186
+ try {
187
+ if (localFileExisted) {
188
+ nativeBinding = require('./holo-tree.linux-x64-gnu.node')
189
+ } else {
190
+ nativeBinding = require('@hologit/holo-tree-linux-x64-gnu')
191
+ }
192
+ } catch (e) {
193
+ loadError = e
194
+ }
195
+ }
196
+ break
197
+ case 'arm64':
198
+ if (isMusl()) {
199
+ localFileExisted = existsSync(
200
+ join(__dirname, 'holo-tree.linux-arm64-musl.node')
201
+ )
202
+ try {
203
+ if (localFileExisted) {
204
+ nativeBinding = require('./holo-tree.linux-arm64-musl.node')
205
+ } else {
206
+ nativeBinding = require('@hologit/holo-tree-linux-arm64-musl')
207
+ }
208
+ } catch (e) {
209
+ loadError = e
210
+ }
211
+ } else {
212
+ localFileExisted = existsSync(
213
+ join(__dirname, 'holo-tree.linux-arm64-gnu.node')
214
+ )
215
+ try {
216
+ if (localFileExisted) {
217
+ nativeBinding = require('./holo-tree.linux-arm64-gnu.node')
218
+ } else {
219
+ nativeBinding = require('@hologit/holo-tree-linux-arm64-gnu')
220
+ }
221
+ } catch (e) {
222
+ loadError = e
223
+ }
224
+ }
225
+ break
226
+ case 'arm':
227
+ if (isMusl()) {
228
+ localFileExisted = existsSync(
229
+ join(__dirname, 'holo-tree.linux-arm-musleabihf.node')
230
+ )
231
+ try {
232
+ if (localFileExisted) {
233
+ nativeBinding = require('./holo-tree.linux-arm-musleabihf.node')
234
+ } else {
235
+ nativeBinding = require('@hologit/holo-tree-linux-arm-musleabihf')
236
+ }
237
+ } catch (e) {
238
+ loadError = e
239
+ }
240
+ } else {
241
+ localFileExisted = existsSync(
242
+ join(__dirname, 'holo-tree.linux-arm-gnueabihf.node')
243
+ )
244
+ try {
245
+ if (localFileExisted) {
246
+ nativeBinding = require('./holo-tree.linux-arm-gnueabihf.node')
247
+ } else {
248
+ nativeBinding = require('@hologit/holo-tree-linux-arm-gnueabihf')
249
+ }
250
+ } catch (e) {
251
+ loadError = e
252
+ }
253
+ }
254
+ break
255
+ case 'riscv64':
256
+ if (isMusl()) {
257
+ localFileExisted = existsSync(
258
+ join(__dirname, 'holo-tree.linux-riscv64-musl.node')
259
+ )
260
+ try {
261
+ if (localFileExisted) {
262
+ nativeBinding = require('./holo-tree.linux-riscv64-musl.node')
263
+ } else {
264
+ nativeBinding = require('@hologit/holo-tree-linux-riscv64-musl')
265
+ }
266
+ } catch (e) {
267
+ loadError = e
268
+ }
269
+ } else {
270
+ localFileExisted = existsSync(
271
+ join(__dirname, 'holo-tree.linux-riscv64-gnu.node')
272
+ )
273
+ try {
274
+ if (localFileExisted) {
275
+ nativeBinding = require('./holo-tree.linux-riscv64-gnu.node')
276
+ } else {
277
+ nativeBinding = require('@hologit/holo-tree-linux-riscv64-gnu')
278
+ }
279
+ } catch (e) {
280
+ loadError = e
281
+ }
282
+ }
283
+ break
284
+ case 's390x':
285
+ localFileExisted = existsSync(
286
+ join(__dirname, 'holo-tree.linux-s390x-gnu.node')
287
+ )
288
+ try {
289
+ if (localFileExisted) {
290
+ nativeBinding = require('./holo-tree.linux-s390x-gnu.node')
291
+ } else {
292
+ nativeBinding = require('@hologit/holo-tree-linux-s390x-gnu')
293
+ }
294
+ } catch (e) {
295
+ loadError = e
296
+ }
297
+ break
298
+ default:
299
+ throw new Error(`Unsupported architecture on Linux: ${arch}`)
300
+ }
301
+ break
302
+ default:
303
+ throw new Error(`Unsupported OS: ${platform}, architecture: ${arch}`)
304
+ }
305
+
306
+ if (!nativeBinding) {
307
+ if (loadError) {
308
+ throw loadError
309
+ }
310
+ throw new Error(`Failed to load native binding`)
311
+ }
312
+
313
+ const { emptyTreeHash, Repo, Tree } = nativeBinding
314
+
315
+ module.exports.emptyTreeHash = emptyTreeHash
316
+ module.exports.Repo = Repo
317
+ module.exports.Tree = Tree
package/package.json ADDED
@@ -0,0 +1,47 @@
1
+ {
2
+ "name": "@hologit/holo-tree",
3
+ "version": "0.0.1",
4
+ "description": "Node.js native binding for holo-tree — mutable in-memory git trees via gix",
5
+ "main": "index.js",
6
+ "types": "index.d.ts",
7
+ "license": "MIT",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "git+https://github.com/JarvusInnovations/hologit.git",
11
+ "directory": "holo-tree-napi"
12
+ },
13
+ "engines": {
14
+ "node": ">=20"
15
+ },
16
+ "napi": {
17
+ "name": "holo-tree",
18
+ "triples": {
19
+ "defaults": false,
20
+ "additional": [
21
+ "x86_64-unknown-linux-gnu",
22
+ "aarch64-apple-darwin",
23
+ "x86_64-pc-windows-msvc"
24
+ ]
25
+ }
26
+ },
27
+ "files": [
28
+ "index.js",
29
+ "index.d.ts"
30
+ ],
31
+ "optionalDependencies": {
32
+ "@hologit/holo-tree-linux-x64-gnu": "0.0.1",
33
+ "@hologit/holo-tree-darwin-arm64": "0.0.1",
34
+ "@hologit/holo-tree-win32-x64-msvc": "0.0.1"
35
+ },
36
+ "scripts": {
37
+ "artifacts": "napi artifacts",
38
+ "build": "napi build --platform --release",
39
+ "build:debug": "napi build --platform",
40
+ "prepublishOnly": "napi prepublish -t npm",
41
+ "test": "node --test test/",
42
+ "version": "napi version"
43
+ },
44
+ "devDependencies": {
45
+ "@napi-rs/cli": "^2.18.4"
46
+ }
47
+ }