@sochdb/sochdb 0.4.0

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 (78) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +3349 -0
  3. package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
  4. package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
  5. package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
  6. package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
  7. package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
  8. package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
  9. package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
  10. package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
  11. package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
  12. package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
  13. package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
  14. package/bin/sochdb-bulk.js +80 -0
  15. package/bin/sochdb-grpc-server.js +80 -0
  16. package/bin/sochdb-server.js +84 -0
  17. package/dist/cjs/analytics.js +196 -0
  18. package/dist/cjs/database.js +929 -0
  19. package/dist/cjs/embedded/database.js +236 -0
  20. package/dist/cjs/embedded/ffi/bindings.js +113 -0
  21. package/dist/cjs/embedded/ffi/library-finder.js +135 -0
  22. package/dist/cjs/embedded/index.js +14 -0
  23. package/dist/cjs/embedded/transaction.js +172 -0
  24. package/dist/cjs/errors.js +71 -0
  25. package/dist/cjs/format.js +176 -0
  26. package/dist/cjs/grpc-client.js +328 -0
  27. package/dist/cjs/index.js +75 -0
  28. package/dist/cjs/ipc-client.js +504 -0
  29. package/dist/cjs/query.js +154 -0
  30. package/dist/cjs/server-manager.js +295 -0
  31. package/dist/cjs/sql-engine.js +874 -0
  32. package/dist/esm/analytics.js +196 -0
  33. package/dist/esm/database.js +931 -0
  34. package/dist/esm/embedded/database.js +239 -0
  35. package/dist/esm/embedded/ffi/bindings.js +142 -0
  36. package/dist/esm/embedded/ffi/library-finder.js +135 -0
  37. package/dist/esm/embedded/index.js +14 -0
  38. package/dist/esm/embedded/transaction.js +176 -0
  39. package/dist/esm/errors.js +71 -0
  40. package/dist/esm/format.js +179 -0
  41. package/dist/esm/grpc-client.js +333 -0
  42. package/dist/esm/index.js +75 -0
  43. package/dist/esm/ipc-client.js +505 -0
  44. package/dist/esm/query.js +159 -0
  45. package/dist/esm/server-manager.js +295 -0
  46. package/dist/esm/sql-engine.js +875 -0
  47. package/dist/types/analytics.d.ts +66 -0
  48. package/dist/types/analytics.d.ts.map +1 -0
  49. package/dist/types/database.d.ts +523 -0
  50. package/dist/types/database.d.ts.map +1 -0
  51. package/dist/types/embedded/database.d.ts +105 -0
  52. package/dist/types/embedded/database.d.ts.map +1 -0
  53. package/dist/types/embedded/ffi/bindings.d.ts +24 -0
  54. package/dist/types/embedded/ffi/bindings.d.ts.map +1 -0
  55. package/dist/types/embedded/ffi/library-finder.d.ts +17 -0
  56. package/dist/types/embedded/ffi/library-finder.d.ts.map +1 -0
  57. package/dist/types/embedded/index.d.ts +9 -0
  58. package/dist/types/embedded/index.d.ts.map +1 -0
  59. package/dist/types/embedded/transaction.d.ts +21 -0
  60. package/dist/types/embedded/transaction.d.ts.map +1 -0
  61. package/dist/types/errors.d.ts +36 -0
  62. package/dist/types/errors.d.ts.map +1 -0
  63. package/dist/types/format.d.ts +117 -0
  64. package/dist/types/format.d.ts.map +1 -0
  65. package/dist/types/grpc-client.d.ts +120 -0
  66. package/dist/types/grpc-client.d.ts.map +1 -0
  67. package/dist/types/index.d.ts +50 -0
  68. package/dist/types/index.d.ts.map +1 -0
  69. package/dist/types/ipc-client.d.ts +177 -0
  70. package/dist/types/ipc-client.d.ts.map +1 -0
  71. package/dist/types/query.d.ts +85 -0
  72. package/dist/types/query.d.ts.map +1 -0
  73. package/dist/types/server-manager.d.ts +29 -0
  74. package/dist/types/server-manager.d.ts.map +1 -0
  75. package/dist/types/sql-engine.d.ts +100 -0
  76. package/dist/types/sql-engine.d.ts.map +1 -0
  77. package/package.json +90 -0
  78. package/scripts/postinstall.js +50 -0
@@ -0,0 +1,172 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.EmbeddedTransaction = void 0;
37
+ const errors_1 = require("../errors");
38
+ const bindings_1 = require("./ffi/bindings");
39
+ const koffi = __importStar(require("koffi"));
40
+ class EmbeddedTransaction {
41
+ constructor(db, dbHandle, txnHandle) {
42
+ this.committed = false;
43
+ this.aborted = false;
44
+ this.db = db;
45
+ this.dbHandle = dbHandle;
46
+ this.txnHandle = txnHandle;
47
+ this.bindings = bindings_1.NativeBindings.getInstance();
48
+ }
49
+ async put(key, value) {
50
+ this.ensureActive();
51
+ const res = this.bindings.sochdb_put(this.dbHandle, this.txnHandle, key, key.length, value, value.length);
52
+ if (res !== 0) {
53
+ throw new errors_1.DatabaseError('Failed to put value');
54
+ }
55
+ }
56
+ async get(key) {
57
+ this.ensureActive();
58
+ const outPtr = [null];
59
+ const outLen = [0];
60
+ // returns 0 on success, 1 on not found, -1 on error
61
+ const res = this.bindings.sochdb_get(this.dbHandle, this.txnHandle, key, key.length, outPtr, outLen);
62
+ if (res === 1) { // Not found
63
+ return null;
64
+ }
65
+ if (res !== 0) {
66
+ throw new errors_1.DatabaseError('Failed to get value');
67
+ }
68
+ // Copy buffer
69
+ const ptr = outPtr[0];
70
+ const len = outLen[0];
71
+ const buffer = Buffer.from(koffi.decode(ptr, 'uint8', len));
72
+ // Free native memory
73
+ this.bindings.sochdb_free_bytes(ptr, len);
74
+ return buffer;
75
+ }
76
+ async delete(key) {
77
+ this.ensureActive();
78
+ const res = this.bindings.sochdb_delete(this.dbHandle, this.txnHandle, key, key.length);
79
+ if (res !== 0) {
80
+ throw new errors_1.DatabaseError('Failed to delete value');
81
+ }
82
+ }
83
+ async putPath(path, value) {
84
+ this.ensureActive();
85
+ const res = this.bindings.sochdb_put_path(this.dbHandle, this.txnHandle, path, value, value.length);
86
+ if (res !== 0) {
87
+ throw new errors_1.DatabaseError('Failed to put path');
88
+ }
89
+ }
90
+ async getPath(path) {
91
+ this.ensureActive();
92
+ const outPtr = [null];
93
+ const outLen = [0];
94
+ const res = this.bindings.sochdb_get_path(this.dbHandle, this.txnHandle, path, outPtr, outLen);
95
+ if (res === 1) {
96
+ return null;
97
+ }
98
+ if (res !== 0) {
99
+ throw new errors_1.DatabaseError('Failed to get path');
100
+ }
101
+ const ptr = outPtr[0];
102
+ const len = outLen[0];
103
+ const buffer = Buffer.from(koffi.decode(ptr, 'uint8', len));
104
+ this.bindings.sochdb_free_bytes(ptr, len);
105
+ return buffer;
106
+ }
107
+ async *scanPrefix(prefix) {
108
+ this.ensureActive();
109
+ const iter = this.bindings.sochdb_scan_prefix(this.dbHandle, this.txnHandle, prefix, prefix.length);
110
+ if (!iter)
111
+ return;
112
+ try {
113
+ const keyPtr = [null];
114
+ const keyLen = [0];
115
+ const valPtr = [null];
116
+ const valLen = [0];
117
+ while (true) {
118
+ // Returns 0 on success, 1 on done, -1 on error
119
+ const res = this.bindings.sochdb_iterator_next(iter, keyPtr, keyLen, valPtr, valLen);
120
+ if (res === 1)
121
+ break; // Done
122
+ if (res !== 0)
123
+ throw new errors_1.DatabaseError('Scan failed');
124
+ // Decode key
125
+ const k = Buffer.from(koffi.decode(keyPtr[0], 'uint8', keyLen[0]));
126
+ // koffi automatically handles pointers, but here we received a pointer value into keyPtr[0]
127
+ // Wait, keyPtr is passed as out(pointer(uint8*)).
128
+ // So keyPtr[0] IS the pointer.
129
+ // Rust scan_next allocates new boxed slices for key and value?
130
+ // Yes: `let mut key_buf = key.into_boxed_slice();`
131
+ // So we own it and must free it?
132
+ // `let _ = Box::into_raw(key_buf);` -> leaks.
133
+ // But `sochdb_scan_next` doesn't seem to export a free function for these individual items?
134
+ // Wait, `sochdb_scan_next` returns them. Does the caller free them?
135
+ // Usually yes. `NativeBindings` has `sochdb_free_bytes`.
136
+ this.bindings.sochdb_free_bytes(keyPtr[0], keyLen[0]);
137
+ // Decode value
138
+ const v = Buffer.from(koffi.decode(valPtr[0], 'uint8', valLen[0]));
139
+ this.bindings.sochdb_free_bytes(valPtr[0], valLen[0]);
140
+ yield [k, v];
141
+ }
142
+ }
143
+ finally {
144
+ this.bindings.sochdb_iterator_close(iter);
145
+ }
146
+ }
147
+ async commit() {
148
+ this.ensureActive();
149
+ const result = this.bindings.sochdb_commit(this.dbHandle, this.txnHandle);
150
+ this.committed = true;
151
+ if (result.error_code !== 0) {
152
+ // -1 indicates error, -2 indicates SSI conflict
153
+ throw new errors_1.TransactionError(`Transaction failed to commit (Code ${result.error_code})`);
154
+ }
155
+ }
156
+ async abort() {
157
+ if (!this.isActive())
158
+ return;
159
+ this.bindings.sochdb_abort(this.dbHandle, this.txnHandle);
160
+ this.aborted = true;
161
+ }
162
+ isActive() {
163
+ return !this.committed && !this.aborted;
164
+ }
165
+ ensureActive() {
166
+ if (!this.isActive()) {
167
+ throw new errors_1.TransactionError('Transaction is no longer active');
168
+ }
169
+ }
170
+ }
171
+ exports.EmbeddedTransaction = EmbeddedTransaction;
172
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/embedded/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransactionError, DatabaseError } from '../errors';\nimport { NativeBindings } from './ffi/bindings';\nimport { EmbeddedDatabase } from './database';\nimport * as koffi from 'koffi';\n\nexport class EmbeddedTransaction {\n    private db: EmbeddedDatabase;\n    private dbHandle: any;\n    private txnHandle: any; // Now a Struct object (C_TxnHandle)\n    private bindings: NativeBindings;\n    private committed = false;\n    private aborted = false;\n\n    constructor(db: EmbeddedDatabase, dbHandle: any, txnHandle: any) {\n        this.db = db;\n        this.dbHandle = dbHandle;\n        this.txnHandle = txnHandle;\n        this.bindings = NativeBindings.getInstance();\n    }\n\n    async put(key: Buffer, value: Buffer): Promise<void> {\n        this.ensureActive();\n        const res = this.bindings.sochdb_put(this.dbHandle, this.txnHandle, key, key.length, value, value.length);\n        if (res !== 0) {\n            throw new DatabaseError('Failed to put value');\n        }\n    }\n\n    async get(key: Buffer): Promise<Buffer | null> {\n        this.ensureActive();\n\n        const outPtr = [null];\n        const outLen = [0];\n\n        // returns 0 on success, 1 on not found, -1 on error\n        const res = this.bindings.sochdb_get(this.dbHandle, this.txnHandle, key, key.length, outPtr, outLen);\n\n        if (res === 1) { // Not found\n            return null;\n        }\n        if (res !== 0) {\n            throw new DatabaseError('Failed to get value');\n        }\n\n        // Copy buffer\n        const ptr = outPtr[0];\n        const len = outLen[0];\n        const buffer = Buffer.from(koffi.decode(ptr, 'uint8', len));\n\n        // Free native memory\n        this.bindings.sochdb_free_bytes(ptr, len);\n\n        return buffer;\n    }\n\n    async delete(key: Buffer): Promise<void> {\n        this.ensureActive();\n        const res = this.bindings.sochdb_delete(this.dbHandle, this.txnHandle, key, key.length);\n        if (res !== 0) {\n            throw new DatabaseError('Failed to delete value');\n        }\n    }\n\n    async putPath(path: string, value: Buffer): Promise<void> {\n        this.ensureActive();\n        const res = this.bindings.sochdb_put_path(this.dbHandle, this.txnHandle, path, value, value.length);\n        if (res !== 0) {\n            throw new DatabaseError('Failed to put path');\n        }\n    }\n\n    async getPath(path: string): Promise<Buffer | null> {\n        this.ensureActive();\n\n        const outPtr = [null];\n        const outLen = [0];\n\n        const res = this.bindings.sochdb_get_path(this.dbHandle, this.txnHandle, path, outPtr, outLen);\n\n        if (res === 1) {\n            return null;\n        }\n        if (res !== 0) {\n            throw new DatabaseError('Failed to get path');\n        }\n\n        const ptr = outPtr[0];\n        const len = outLen[0];\n        const buffer = Buffer.from(koffi.decode(ptr, 'uint8', len));\n        this.bindings.sochdb_free_bytes(ptr, len);\n\n        return buffer;\n    }\n\n    async *scanPrefix(prefix: Buffer): AsyncGenerator<[Buffer, Buffer]> {\n        this.ensureActive();\n\n        const iter = this.bindings.sochdb_scan_prefix(this.dbHandle, this.txnHandle, prefix, prefix.length);\n        if (!iter) return;\n\n        try {\n            const keyPtr = [null];\n            const keyLen = [0];\n            const valPtr = [null];\n            const valLen = [0];\n\n            while (true) {\n                // Returns 0 on success, 1 on done, -1 on error\n                const res = this.bindings.sochdb_iterator_next(iter, keyPtr, keyLen, valPtr, valLen);\n                if (res === 1) break; // Done\n                if (res !== 0) throw new DatabaseError('Scan failed');\n\n                // Decode key\n                const k = Buffer.from(koffi.decode(keyPtr[0], 'uint8', keyLen[0]));\n                // koffi automatically handles pointers, but here we received a pointer value into keyPtr[0]\n                // Wait, keyPtr is passed as out(pointer(uint8*)).\n                // So keyPtr[0] IS the pointer.\n\n                // Rust scan_next allocates new boxed slices for key and value?\n                // Yes: `let mut key_buf = key.into_boxed_slice();`\n                // So we own it and must free it?\n                // `let _ = Box::into_raw(key_buf);` -> leaks.\n                // But `sochdb_scan_next` doesn't seem to export a free function for these individual items?\n                // Wait, `sochdb_scan_next` returns them. Does the caller free them?\n                // Usually yes. `NativeBindings` has `sochdb_free_bytes`.\n\n                this.bindings.sochdb_free_bytes(keyPtr[0], keyLen[0]);\n\n                // Decode value \n                const v = Buffer.from(koffi.decode(valPtr[0], 'uint8', valLen[0]));\n                this.bindings.sochdb_free_bytes(valPtr[0], valLen[0]);\n\n                yield [k, v];\n            }\n        } finally {\n            this.bindings.sochdb_iterator_close(iter);\n        }\n    }\n\n    async commit(): Promise<void> {\n        this.ensureActive();\n\n        const result = this.bindings.sochdb_commit(this.dbHandle, this.txnHandle);\n        this.committed = true;\n\n        if (result.error_code !== 0) {\n            // -1 indicates error, -2 indicates SSI conflict\n            throw new TransactionError(`Transaction failed to commit (Code ${result.error_code})`);\n        }\n    }\n\n    async abort(): Promise<void> {\n        if (!this.isActive()) return;\n\n        this.bindings.sochdb_abort(this.dbHandle, this.txnHandle);\n        this.aborted = true;\n    }\n\n    private isActive(): boolean {\n        return !this.committed && !this.aborted;\n    }\n\n    private ensureActive(): void {\n        if (!this.isActive()) {\n            throw new TransactionError('Transaction is no longer active');\n        }\n    }\n}\n"]}
@@ -0,0 +1,71 @@
1
+ "use strict";
2
+ /**
3
+ * SochDB Error Classes
4
+ *
5
+ * @packageDocumentation
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.DatabaseError = exports.ProtocolError = exports.TransactionError = exports.ConnectionError = exports.SochDBError = void 0;
9
+ // Copyright 2025 Sushanth (https://github.com/sushanthpy)
10
+ //
11
+ // Licensed under the Apache License, Version 2.0 (the "License");
12
+ // you may not use this file except in compliance with the License.
13
+ // You may obtain a copy of the License at
14
+ //
15
+ // http://www.apache.org/licenses/LICENSE-2.0
16
+ /**
17
+ * Base error class for all SochDB errors.
18
+ */
19
+ class SochDBError extends Error {
20
+ constructor(message) {
21
+ super(message);
22
+ this.name = 'SochDBError';
23
+ Object.setPrototypeOf(this, SochDBError.prototype);
24
+ }
25
+ }
26
+ exports.SochDBError = SochDBError;
27
+ /**
28
+ * Error thrown when connection to the database fails.
29
+ */
30
+ class ConnectionError extends SochDBError {
31
+ constructor(message) {
32
+ super(message);
33
+ this.name = 'ConnectionError';
34
+ Object.setPrototypeOf(this, ConnectionError.prototype);
35
+ }
36
+ }
37
+ exports.ConnectionError = ConnectionError;
38
+ /**
39
+ * Error thrown when a transaction operation fails.
40
+ */
41
+ class TransactionError extends SochDBError {
42
+ constructor(message) {
43
+ super(message);
44
+ this.name = 'TransactionError';
45
+ Object.setPrototypeOf(this, TransactionError.prototype);
46
+ }
47
+ }
48
+ exports.TransactionError = TransactionError;
49
+ /**
50
+ * Error thrown when there's a protocol error in IPC communication.
51
+ */
52
+ class ProtocolError extends SochDBError {
53
+ constructor(message) {
54
+ super(message);
55
+ this.name = 'ProtocolError';
56
+ Object.setPrototypeOf(this, ProtocolError.prototype);
57
+ }
58
+ }
59
+ exports.ProtocolError = ProtocolError;
60
+ /**
61
+ * Error thrown when a database operation fails.
62
+ */
63
+ class DatabaseError extends SochDBError {
64
+ constructor(message) {
65
+ super(message);
66
+ this.name = 'DatabaseError';
67
+ Object.setPrototypeOf(this, DatabaseError.prototype);
68
+ }
69
+ }
70
+ exports.DatabaseError = DatabaseError;
71
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7O0FBRUgsMERBQTBEO0FBQzFELEVBQUU7QUFDRixrRUFBa0U7QUFDbEUsbUVBQW1FO0FBQ25FLDBDQUEwQztBQUMxQyxFQUFFO0FBQ0YsaURBQWlEO0FBRWpEOztHQUVHO0FBQ0gsTUFBYSxXQUFZLFNBQVEsS0FBSztJQUNwQyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxhQUFhLENBQUM7UUFDMUIsTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ3JELENBQUM7Q0FDRjtBQU5ELGtDQU1DO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLGVBQWdCLFNBQVEsV0FBVztJQUM5QyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxpQkFBaUIsQ0FBQztRQUM5QixNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxlQUFlLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDekQsQ0FBQztDQUNGO0FBTkQsMENBTUM7QUFFRDs7R0FFRztBQUNILE1BQWEsZ0JBQWlCLFNBQVEsV0FBVztJQUMvQyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxrQkFBa0IsQ0FBQztRQUMvQixNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxnQkFBZ0IsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUMxRCxDQUFDO0NBQ0Y7QUFORCw0Q0FNQztBQUVEOztHQUVHO0FBQ0gsTUFBYSxhQUFjLFNBQVEsV0FBVztJQUM1QyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2YsSUFBSSxDQUFDLElBQUksR0FBRyxlQUFlLENBQUM7UUFDNUIsTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsYUFBYSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ3ZELENBQUM7Q0FDRjtBQU5ELHNDQU1DO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLGFBQWMsU0FBUSxXQUFXO0lBQzVDLFlBQVksT0FBZTtRQUN6QixLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDZixJQUFJLENBQUMsSUFBSSxHQUFHLGVBQWUsQ0FBQztRQUM1QixNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxhQUFhLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDdkQsQ0FBQztDQUNGO0FBTkQsc0NBTUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNvY2hEQiBFcnJvciBDbGFzc2VzXG4gKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKi9cblxuLy8gQ29weXJpZ2h0IDIwMjUgU3VzaGFudGggKGh0dHBzOi8vZ2l0aHViLmNvbS9zdXNoYW50aHB5KVxuLy9cbi8vIExpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSBcIkxpY2Vuc2VcIik7XG4vLyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuXG4vLyBZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXRcbi8vXG4vLyAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wXG5cbi8qKlxuICogQmFzZSBlcnJvciBjbGFzcyBmb3IgYWxsIFNvY2hEQiBlcnJvcnMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTb2NoREJFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IobWVzc2FnZTogc3RyaW5nKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gJ1NvY2hEQkVycm9yJztcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2YodGhpcywgU29jaERCRXJyb3IucHJvdG90eXBlKTtcbiAgfVxufVxuXG4vKipcbiAqIEVycm9yIHRocm93biB3aGVuIGNvbm5lY3Rpb24gdG8gdGhlIGRhdGFiYXNlIGZhaWxzLlxuICovXG5leHBvcnQgY2xhc3MgQ29ubmVjdGlvbkVycm9yIGV4dGVuZHMgU29jaERCRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihtZXNzYWdlOiBzdHJpbmcpIHtcbiAgICBzdXBlcihtZXNzYWdlKTtcbiAgICB0aGlzLm5hbWUgPSAnQ29ubmVjdGlvbkVycm9yJztcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2YodGhpcywgQ29ubmVjdGlvbkVycm9yLnByb3RvdHlwZSk7XG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBhIHRyYW5zYWN0aW9uIG9wZXJhdGlvbiBmYWlscy5cbiAqL1xuZXhwb3J0IGNsYXNzIFRyYW5zYWN0aW9uRXJyb3IgZXh0ZW5kcyBTb2NoREJFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKG1lc3NhZ2UpO1xuICAgIHRoaXMubmFtZSA9ICdUcmFuc2FjdGlvbkVycm9yJztcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2YodGhpcywgVHJhbnNhY3Rpb25FcnJvci5wcm90b3R5cGUpO1xuICB9XG59XG5cbi8qKlxuICogRXJyb3IgdGhyb3duIHdoZW4gdGhlcmUncyBhIHByb3RvY29sIGVycm9yIGluIElQQyBjb21tdW5pY2F0aW9uLlxuICovXG5leHBvcnQgY2xhc3MgUHJvdG9jb2xFcnJvciBleHRlbmRzIFNvY2hEQkVycm9yIHtcbiAgY29uc3RydWN0b3IobWVzc2FnZTogc3RyaW5nKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gJ1Byb3RvY29sRXJyb3InO1xuICAgIE9iamVjdC5zZXRQcm90b3R5cGVPZih0aGlzLCBQcm90b2NvbEVycm9yLnByb3RvdHlwZSk7XG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBhIGRhdGFiYXNlIG9wZXJhdGlvbiBmYWlscy5cbiAqL1xuZXhwb3J0IGNsYXNzIERhdGFiYXNlRXJyb3IgZXh0ZW5kcyBTb2NoREJFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKG1lc3NhZ2UpO1xuICAgIHRoaXMubmFtZSA9ICdEYXRhYmFzZUVycm9yJztcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2YodGhpcywgRGF0YWJhc2VFcnJvci5wcm90b3R5cGUpO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,176 @@
1
+ "use strict";
2
+ /**
3
+ * Copyright 2025 Sushanth (https://github.com/sushanthpy)
4
+ *
5
+ * Licensed under the Apache License, Version 2.0 (the "License");
6
+ * you may not use this file except in compliance with the License.
7
+ * You may obtain a copy of the License at
8
+ *
9
+ * http://www.apache.org/licenses/LICENSE-2.0
10
+ *
11
+ * Unless required by applicable law or agreed to in writing, software
12
+ * distributed under the License is distributed on an "AS IS" BASIS,
13
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
+ * See the License for the specific language governing permissions and
15
+ * limitations under the License.
16
+ */
17
+ Object.defineProperty(exports, "__esModule", { value: true });
18
+ exports.FormatCapabilities = exports.CanonicalFormat = exports.ContextFormat = exports.WireFormat = exports.FormatConversionError = void 0;
19
+ /**
20
+ * Unified Output Format Semantics
21
+ *
22
+ * Provides format enums for query results and LLM context packaging.
23
+ * This mirrors the Rust sochdb-client format module for consistency.
24
+ */
25
+ /**
26
+ * Error when format conversion fails.
27
+ */
28
+ class FormatConversionError extends Error {
29
+ constructor(fromFormat, toFormat, reason) {
30
+ super(`Cannot convert ${fromFormat} to ${toFormat}: ${reason}`);
31
+ this.fromFormat = fromFormat;
32
+ this.toFormat = toFormat;
33
+ this.reason = reason;
34
+ this.name = 'FormatConversionError';
35
+ }
36
+ }
37
+ exports.FormatConversionError = FormatConversionError;
38
+ /**
39
+ * Output format for query results sent to clients.
40
+ *
41
+ * These formats are optimized for transmission efficiency and
42
+ * client-side processing.
43
+ */
44
+ var WireFormat;
45
+ (function (WireFormat) {
46
+ /**
47
+ * TOON format (default, 40-66% fewer tokens than JSON).
48
+ * Optimized for LLM consumption.
49
+ */
50
+ WireFormat["TOON"] = "toon";
51
+ /**
52
+ * Standard JSON for compatibility.
53
+ */
54
+ WireFormat["JSON"] = "json";
55
+ /**
56
+ * Raw columnar format for analytics.
57
+ * More efficient for large result sets with projection pushdown.
58
+ */
59
+ WireFormat["COLUMNAR"] = "columnar";
60
+ })(WireFormat || (exports.WireFormat = WireFormat = {}));
61
+ (function (WireFormat) {
62
+ /**
63
+ * Parse format from string.
64
+ */
65
+ function fromString(s) {
66
+ const lower = s.toLowerCase();
67
+ switch (lower) {
68
+ case 'toon':
69
+ return WireFormat.TOON;
70
+ case 'json':
71
+ return WireFormat.JSON;
72
+ case 'columnar':
73
+ case 'column':
74
+ return WireFormat.COLUMNAR;
75
+ default:
76
+ throw new FormatConversionError(s, 'WireFormat', `Unknown format '${s}'. Valid: toon, json, columnar`);
77
+ }
78
+ }
79
+ WireFormat.fromString = fromString;
80
+ })(WireFormat || (exports.WireFormat = WireFormat = {}));
81
+ /**
82
+ * Output format for LLM context packaging.
83
+ *
84
+ * These formats are optimized for readability and token efficiency
85
+ * when constructing prompts for language models.
86
+ */
87
+ var ContextFormat;
88
+ (function (ContextFormat) {
89
+ /**
90
+ * TOON format (default, token-efficient).
91
+ * Structured data with minimal syntax overhead.
92
+ */
93
+ ContextFormat["TOON"] = "toon";
94
+ /**
95
+ * JSON format.
96
+ * Widely understood by LLMs, good for structured data.
97
+ */
98
+ ContextFormat["JSON"] = "json";
99
+ /**
100
+ * Markdown format.
101
+ * Best for human-readable context with formatting.
102
+ */
103
+ ContextFormat["MARKDOWN"] = "markdown";
104
+ })(ContextFormat || (exports.ContextFormat = ContextFormat = {}));
105
+ (function (ContextFormat) {
106
+ /**
107
+ * Parse format from string.
108
+ */
109
+ function fromString(s) {
110
+ const lower = s.toLowerCase();
111
+ switch (lower) {
112
+ case 'toon':
113
+ return ContextFormat.TOON;
114
+ case 'json':
115
+ return ContextFormat.JSON;
116
+ case 'markdown':
117
+ case 'md':
118
+ return ContextFormat.MARKDOWN;
119
+ default:
120
+ throw new FormatConversionError(s, 'ContextFormat', `Unknown format '${s}'. Valid: toon, json, markdown`);
121
+ }
122
+ }
123
+ ContextFormat.fromString = fromString;
124
+ })(ContextFormat || (exports.ContextFormat = ContextFormat = {}));
125
+ /**
126
+ * Canonical storage format (server-side only).
127
+ *
128
+ * This is the format used for internal storage and is optimized
129
+ * for storage efficiency and query performance.
130
+ */
131
+ var CanonicalFormat;
132
+ (function (CanonicalFormat) {
133
+ /**
134
+ * TOON canonical format.
135
+ */
136
+ CanonicalFormat["TOON"] = "toon";
137
+ })(CanonicalFormat || (exports.CanonicalFormat = CanonicalFormat = {}));
138
+ /**
139
+ * Helper to check format capabilities and conversions.
140
+ */
141
+ class FormatCapabilities {
142
+ /**
143
+ * Convert WireFormat to ContextFormat if compatible.
144
+ */
145
+ static wireToContext(wire) {
146
+ switch (wire) {
147
+ case WireFormat.TOON:
148
+ return ContextFormat.TOON;
149
+ case WireFormat.JSON:
150
+ return ContextFormat.JSON;
151
+ default:
152
+ return null;
153
+ }
154
+ }
155
+ /**
156
+ * Convert ContextFormat to WireFormat if compatible.
157
+ */
158
+ static contextToWire(ctx) {
159
+ switch (ctx) {
160
+ case ContextFormat.TOON:
161
+ return WireFormat.TOON;
162
+ case ContextFormat.JSON:
163
+ return WireFormat.JSON;
164
+ default:
165
+ return null;
166
+ }
167
+ }
168
+ /**
169
+ * Check if format supports round-trip: decode(encode(x)) = x.
170
+ */
171
+ static supportsRoundTrip(fmt) {
172
+ return fmt === WireFormat.TOON || fmt === WireFormat.JSON;
173
+ }
174
+ }
175
+ exports.FormatCapabilities = FormatCapabilities;
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"format.js","sourceRoot":"","sources":["../../src/format.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright 2025 Sushanth (https://github.com/sushanthpy)\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Unified Output Format Semantics\n * \n * Provides format enums for query results and LLM context packaging.\n * This mirrors the Rust sochdb-client format module for consistency.\n */\n\n/**\n * Error when format conversion fails.\n */\nexport class FormatConversionError extends Error {\n  constructor(\n    public fromFormat: string,\n    public toFormat: string,\n    public reason: string\n  ) {\n    super(`Cannot convert ${fromFormat} to ${toFormat}: ${reason}`);\n    this.name = 'FormatConversionError';\n  }\n}\n\n/**\n * Output format for query results sent to clients.\n * \n * These formats are optimized for transmission efficiency and\n * client-side processing.\n */\nexport enum WireFormat {\n  /**\n   * TOON format (default, 40-66% fewer tokens than JSON).\n   * Optimized for LLM consumption.\n   */\n  TOON = 'toon',\n\n  /**\n   * Standard JSON for compatibility.\n   */\n  JSON = 'json',\n\n  /**\n   * Raw columnar format for analytics.\n   * More efficient for large result sets with projection pushdown.\n   */\n  COLUMNAR = 'columnar',\n}\n\nexport namespace WireFormat {\n  /**\n   * Parse format from string.\n   */\n  export function fromString(s: string): WireFormat {\n    const lower = s.toLowerCase();\n    switch (lower) {\n      case 'toon':\n        return WireFormat.TOON;\n      case 'json':\n        return WireFormat.JSON;\n      case 'columnar':\n      case 'column':\n        return WireFormat.COLUMNAR;\n      default:\n        throw new FormatConversionError(\n          s,\n          'WireFormat',\n          `Unknown format '${s}'. Valid: toon, json, columnar`\n        );\n    }\n  }\n}\n\n/**\n * Output format for LLM context packaging.\n * \n * These formats are optimized for readability and token efficiency\n * when constructing prompts for language models.\n */\nexport enum ContextFormat {\n  /**\n   * TOON format (default, token-efficient).\n   * Structured data with minimal syntax overhead.\n   */\n  TOON = 'toon',\n\n  /**\n   * JSON format.\n   * Widely understood by LLMs, good for structured data.\n   */\n  JSON = 'json',\n\n  /**\n   * Markdown format.\n   * Best for human-readable context with formatting.\n   */\n  MARKDOWN = 'markdown',\n}\n\nexport namespace ContextFormat {\n  /**\n   * Parse format from string.\n   */\n  export function fromString(s: string): ContextFormat {\n    const lower = s.toLowerCase();\n    switch (lower) {\n      case 'toon':\n        return ContextFormat.TOON;\n      case 'json':\n        return ContextFormat.JSON;\n      case 'markdown':\n      case 'md':\n        return ContextFormat.MARKDOWN;\n      default:\n        throw new FormatConversionError(\n          s,\n          'ContextFormat',\n          `Unknown format '${s}'. Valid: toon, json, markdown`\n        );\n    }\n  }\n}\n\n/**\n * Canonical storage format (server-side only).\n * \n * This is the format used for internal storage and is optimized\n * for storage efficiency and query performance.\n */\nexport enum CanonicalFormat {\n  /**\n   * TOON canonical format.\n   */\n  TOON = 'toon',\n}\n\n/**\n * Helper to check format capabilities and conversions.\n */\nexport class FormatCapabilities {\n  /**\n   * Convert WireFormat to ContextFormat if compatible.\n   */\n  static wireToContext(wire: WireFormat): ContextFormat | null {\n    switch (wire) {\n      case WireFormat.TOON:\n        return ContextFormat.TOON;\n      case WireFormat.JSON:\n        return ContextFormat.JSON;\n      default:\n        return null;\n    }\n  }\n\n  /**\n   * Convert ContextFormat to WireFormat if compatible.\n   */\n  static contextToWire(ctx: ContextFormat): WireFormat | null {\n    switch (ctx) {\n      case ContextFormat.TOON:\n        return WireFormat.TOON;\n      case ContextFormat.JSON:\n        return WireFormat.JSON;\n      default:\n        return null;\n    }\n  }\n\n  /**\n   * Check if format supports round-trip: decode(encode(x)) = x.\n   */\n  static supportsRoundTrip(fmt: WireFormat): boolean {\n    return fmt === WireFormat.TOON || fmt === WireFormat.JSON;\n  }\n}\n"]}