@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,80 @@
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * SochDB Bulk CLI Wrapper
5
+ * Automatically locates and runs the platform-specific binary.
6
+ */
7
+
8
+ const os = require('os');
9
+ const path = require('path');
10
+ const { spawn } = require('child_process');
11
+ const fs = require('fs');
12
+
13
+ function getPlatformBinary() {
14
+ const platform = os.platform();
15
+ const arch = os.arch();
16
+
17
+ let target = '';
18
+ let ext = '';
19
+
20
+ if (platform === 'darwin') {
21
+ target = arch === 'arm64' ? 'aarch64-apple-darwin' : 'x86_64-apple-darwin';
22
+ } else if (platform === 'linux') {
23
+ target = arch === 'arm64' ? 'aarch64-unknown-linux-gnu' : 'x86_64-unknown-linux-gnu';
24
+ } else if (platform === 'win32') {
25
+ target = 'x86_64-pc-windows-msvc';
26
+ ext = '.exe';
27
+ } else {
28
+ throw new Error(`Unsupported platform: ${platform} ${arch}`);
29
+ }
30
+
31
+ // Look for bundled binary
32
+ const binName = `sochdb-bulk${ext}`;
33
+ const bundledPath = path.resolve(__dirname, '..', '_bin', target, binName);
34
+
35
+ if (fs.existsSync(bundledPath)) {
36
+ return bundledPath;
37
+ }
38
+
39
+ // Fallback: Check environment variable
40
+ if (process.env.SOCHDB_BULK_PATH) {
41
+ return process.env.SOCHDB_BULK_PATH;
42
+ }
43
+
44
+ throw new Error(`
45
+ sochdb-bulk binary not found!
46
+ Searched at: ${bundledPath}
47
+
48
+ To fix:
49
+ 1. Reinstall the package: npm install --force @sushanth/sochdb
50
+ 2. Or set SOCHDB_BULK_PATH environment variable
51
+ `);
52
+ }
53
+
54
+ try {
55
+ const binPath = getPlatformBinary();
56
+
57
+ const child = spawn(binPath, process.argv.slice(2), {
58
+ stdio: 'inherit'
59
+ });
60
+
61
+ child.on('exit', (code) => {
62
+ process.exit(code || 0);
63
+ });
64
+
65
+ child.on('error', (err) => {
66
+ console.error(`Failed to start sochdb-bulk: ${err.message}`);
67
+ process.exit(1);
68
+ });
69
+
70
+ const signals = ['SIGINT', 'SIGTERM'];
71
+ signals.forEach(signal => {
72
+ process.on(signal, () => {
73
+ child.kill(signal);
74
+ });
75
+ });
76
+
77
+ } catch (err) {
78
+ console.error(`[sochdb-js] Error: ${err.message}`);
79
+ process.exit(1);
80
+ }
@@ -0,0 +1,80 @@
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * SochDB gRPC Server CLI Wrapper
5
+ * Automatically locates and runs the platform-specific binary.
6
+ */
7
+
8
+ const os = require('os');
9
+ const path = require('path');
10
+ const { spawn } = require('child_process');
11
+ const fs = require('fs');
12
+
13
+ function getPlatformBinary() {
14
+ const platform = os.platform();
15
+ const arch = os.arch();
16
+
17
+ let target = '';
18
+ let ext = '';
19
+
20
+ if (platform === 'darwin') {
21
+ target = arch === 'arm64' ? 'aarch64-apple-darwin' : 'x86_64-apple-darwin';
22
+ } else if (platform === 'linux') {
23
+ target = arch === 'arm64' ? 'aarch64-unknown-linux-gnu' : 'x86_64-unknown-linux-gnu';
24
+ } else if (platform === 'win32') {
25
+ target = 'x86_64-pc-windows-msvc';
26
+ ext = '.exe';
27
+ } else {
28
+ throw new Error(`Unsupported platform: ${platform} ${arch}`);
29
+ }
30
+
31
+ // Look for bundled binary
32
+ const binName = `sochdb-grpc-server${ext}`;
33
+ const bundledPath = path.resolve(__dirname, '..', '_bin', target, binName);
34
+
35
+ if (fs.existsSync(bundledPath)) {
36
+ return bundledPath;
37
+ }
38
+
39
+ // Fallback: Check environment variable
40
+ if (process.env.SOCHDB_GRPC_SERVER_PATH) {
41
+ return process.env.SOCHDB_GRPC_SERVER_PATH;
42
+ }
43
+
44
+ throw new Error(`
45
+ sochdb-grpc-server binary not found!
46
+ Searched at: ${bundledPath}
47
+
48
+ To fix:
49
+ 1. Reinstall the package: npm install --force @sushanth/sochdb
50
+ 2. Or set SOCHDB_GRPC_SERVER_PATH environment variable
51
+ `);
52
+ }
53
+
54
+ try {
55
+ const binPath = getPlatformBinary();
56
+
57
+ const child = spawn(binPath, process.argv.slice(2), {
58
+ stdio: 'inherit'
59
+ });
60
+
61
+ child.on('exit', (code) => {
62
+ process.exit(code || 0);
63
+ });
64
+
65
+ child.on('error', (err) => {
66
+ console.error(`Failed to start sochdb-grpc-server: ${err.message}`);
67
+ process.exit(1);
68
+ });
69
+
70
+ const signals = ['SIGINT', 'SIGTERM', 'SIGHUP'];
71
+ signals.forEach(signal => {
72
+ process.on(signal, () => {
73
+ child.kill(signal);
74
+ });
75
+ });
76
+
77
+ } catch (err) {
78
+ console.error(`[sochdb-js] Error: ${err.message}`);
79
+ process.exit(1);
80
+ }
@@ -0,0 +1,84 @@
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * SochDB Server CLI Wrapper
5
+ * Automatically locates and runs the platform-specific binary.
6
+ */
7
+
8
+ const os = require('os');
9
+ const path = require('path');
10
+ const { spawn } = require('child_process');
11
+ const fs = require('fs');
12
+
13
+ function getPlatformBinary() {
14
+ const platform = os.platform();
15
+ const arch = os.arch();
16
+
17
+ let target = '';
18
+ let ext = '';
19
+
20
+ if (platform === 'darwin') {
21
+ target = arch === 'arm64' ? 'aarch64-apple-darwin' : 'x86_64-apple-darwin';
22
+ } else if (platform === 'linux') {
23
+ target = arch === 'arm64' ? 'aarch64-unknown-linux-gnu' : 'x86_64-unknown-linux-gnu';
24
+ } else if (platform === 'win32') {
25
+ target = 'x86_64-pc-windows-msvc';
26
+ ext = '.exe';
27
+ } else {
28
+ throw new Error(`Unsupported platform: ${platform} ${arch}`);
29
+ }
30
+
31
+ // Look for bundled binary
32
+ const binName = `sochdb-server${ext}`;
33
+ const bundledPath = path.resolve(__dirname, '..', '_bin', target, binName);
34
+
35
+ if (fs.existsSync(bundledPath)) {
36
+ return bundledPath;
37
+ }
38
+
39
+ // Fallback: Check environment variable
40
+ if (process.env.SOCHDB_SERVER_PATH) {
41
+ return process.env.SOCHDB_SERVER_PATH;
42
+ }
43
+
44
+ throw new Error(`
45
+ sochdb-server binary not found!
46
+ Searched at: ${bundledPath}
47
+
48
+ To fix:
49
+ 1. Reinstall the package: npm install --force @sushanth/sochdb
50
+ 2. Or set SOCHDB_SERVER_PATH environment variable
51
+ `);
52
+ }
53
+
54
+ try {
55
+ const binPath = getPlatformBinary();
56
+
57
+ // Spawn with inherited stdio
58
+ const child = spawn(binPath, process.argv.slice(2), {
59
+ stdio: 'inherit'
60
+ });
61
+
62
+ child.on('exit', (code) => {
63
+ process.exit(code || 0);
64
+ });
65
+
66
+ child.on('error', (err) => {
67
+ console.error(`Failed to start sochdb-server: ${err.message}`);
68
+ process.exit(1);
69
+ });
70
+
71
+ // Handle signals
72
+ const signals = ['SIGINT', 'SIGTERM', 'SIGHUP'];
73
+ signals.forEach(signal => {
74
+ process.on(signal, () => {
75
+ if (child.kill(signal)) {
76
+ // Wait specifically for child exit
77
+ }
78
+ });
79
+ });
80
+
81
+ } catch (err) {
82
+ console.error(`[sochdb-js] Error: ${err.message}`);
83
+ process.exit(1);
84
+ }
@@ -0,0 +1,196 @@
1
+ "use strict";
2
+ /**
3
+ * SochDB Analytics - Anonymous usage tracking with PostHog
4
+ *
5
+ * This module provides anonymous, privacy-respecting analytics to help
6
+ * improve SochDB. All tracking can be disabled by setting:
7
+ *
8
+ * SOCHDB_DISABLE_ANALYTICS=true
9
+ *
10
+ * No personally identifiable information (PII) is collected. Only aggregate
11
+ * usage patterns are tracked to understand:
12
+ * - Which features are most used
13
+ * - Performance characteristics
14
+ * - Error patterns for debugging
15
+ *
16
+ * Copyright 2025 Sushanth (https://github.com/sushanthpy)
17
+ * Licensed under the Apache License, Version 2.0
18
+ */
19
+ Object.defineProperty(exports, "__esModule", { value: true });
20
+ exports.isAnalyticsDisabled = isAnalyticsDisabled;
21
+ exports.capture = capture;
22
+ exports.captureError = captureError;
23
+ exports.shutdown = shutdown;
24
+ exports.trackDatabaseOpen = trackDatabaseOpen;
25
+ const crypto_1 = require("crypto");
26
+ const os_1 = require("os");
27
+ // PostHog configuration
28
+ const POSTHOG_API_KEY = "phc_zf0hm6ZmPUJj1pM07Kigqvphh1ClhKX1NahRU4G0bfu";
29
+ const POSTHOG_HOST = "https://us.i.posthog.com";
30
+ // Lazy-loaded PostHog client
31
+ let posthogClient = null;
32
+ let posthogInitialized = false;
33
+ /**
34
+ * Check if analytics is disabled via environment variable.
35
+ *
36
+ * Analytics is disabled when SOCHDB_DISABLE_ANALYTICS is set to 'true', '1', 'yes', or 'on'.
37
+ *
38
+ * @returns true if analytics is disabled
39
+ */
40
+ function isAnalyticsDisabled() {
41
+ const disableVar = (process.env.SOCHDB_DISABLE_ANALYTICS || "").toLowerCase();
42
+ return ["true", "1", "yes", "on"].includes(disableVar);
43
+ }
44
+ /**
45
+ * Generate a stable anonymous ID for this machine.
46
+ *
47
+ * Uses a hash of machine-specific but non-identifying information.
48
+ * The same machine will always get the same ID, but the ID cannot
49
+ * be reversed to identify the machine.
50
+ */
51
+ function getAnonymousId() {
52
+ try {
53
+ const machineInfo = [
54
+ (0, os_1.hostname)(),
55
+ (0, os_1.platform)(),
56
+ (0, os_1.arch)(),
57
+ process.getuid?.() ?? "windows",
58
+ ].join("|");
59
+ return (0, crypto_1.createHash)("sha256").update(machineInfo).digest("hex").slice(0, 16);
60
+ }
61
+ catch {
62
+ return "anonymous";
63
+ }
64
+ }
65
+ /**
66
+ * Lazily initialize PostHog client.
67
+ */
68
+ async function getPosthogClient() {
69
+ if (isAnalyticsDisabled()) {
70
+ return null;
71
+ }
72
+ if (posthogInitialized) {
73
+ return posthogClient;
74
+ }
75
+ posthogInitialized = true;
76
+ try {
77
+ // Use require to avoid TypeScript checking the optional module
78
+ // This allows the SDK to work without posthog-node installed
79
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
80
+ const posthogModule = require("posthog-node");
81
+ const { PostHog } = posthogModule;
82
+ posthogClient = new PostHog(POSTHOG_API_KEY, {
83
+ host: POSTHOG_HOST,
84
+ });
85
+ return posthogClient;
86
+ }
87
+ catch {
88
+ // posthog-node not installed or error - analytics disabled
89
+ return null;
90
+ }
91
+ }
92
+ /**
93
+ * Get the SDK version.
94
+ */
95
+ function getSdkVersion() {
96
+ try {
97
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
98
+ const pkg = require("../package.json");
99
+ return pkg.version || "unknown";
100
+ }
101
+ catch {
102
+ return "unknown";
103
+ }
104
+ }
105
+ /**
106
+ * Capture an analytics event.
107
+ *
108
+ * This function is a no-op if:
109
+ * - SOCHDB_DISABLE_ANALYTICS=true
110
+ * - posthog-node package is not installed
111
+ * - Any error occurs (fails silently)
112
+ *
113
+ * @param event - Event name (e.g., "database_opened", "vector_search")
114
+ * @param properties - Optional event properties
115
+ * @param distinctId - Optional distinct ID (defaults to anonymous machine ID)
116
+ */
117
+ async function capture(event, properties, distinctId) {
118
+ if (isAnalyticsDisabled()) {
119
+ return;
120
+ }
121
+ try {
122
+ const client = await getPosthogClient();
123
+ if (!client) {
124
+ return;
125
+ }
126
+ // Build properties with SDK context
127
+ const eventProperties = {
128
+ sdk: "nodejs",
129
+ sdk_version: getSdkVersion(),
130
+ node_version: process.version,
131
+ os: (0, os_1.platform)(),
132
+ arch: (0, os_1.arch)(),
133
+ ...properties,
134
+ };
135
+ client.capture({
136
+ distinctId: distinctId || getAnonymousId(),
137
+ event,
138
+ properties: eventProperties,
139
+ });
140
+ // Flush to ensure event is sent immediately
141
+ await client.flush();
142
+ }
143
+ catch {
144
+ // Never let analytics break user code
145
+ }
146
+ }
147
+ /**
148
+ * Capture an error event for debugging.
149
+ *
150
+ * Only sends static information - no dynamic error messages.
151
+ *
152
+ * @param errorType - Static error category (e.g., "connection_error", "query_error", "timeout_error")
153
+ * @param location - Static code location (e.g., "database.open", "query.execute", "transaction.commit")
154
+ * @param properties - Optional additional static properties only
155
+ *
156
+ * @example
157
+ * ```typescript
158
+ * captureError('connection_error', 'database.open');
159
+ * captureError('query_error', 'sql.execute', { query_type: 'SELECT' });
160
+ * ```
161
+ */
162
+ async function captureError(errorType, location, properties) {
163
+ await capture("error", {
164
+ error_type: errorType,
165
+ location,
166
+ ...properties,
167
+ });
168
+ }
169
+ /**
170
+ * Flush any pending events and shutdown the client.
171
+ */
172
+ async function shutdown() {
173
+ if (isAnalyticsDisabled()) {
174
+ return;
175
+ }
176
+ try {
177
+ if (posthogClient) {
178
+ await posthogClient.shutdown();
179
+ }
180
+ }
181
+ catch {
182
+ // Ignore shutdown errors
183
+ }
184
+ }
185
+ // Convenience functions for common events
186
+ /**
187
+ * Track database open event.
188
+ */
189
+ async function trackDatabaseOpen(dbPath, mode = "embedded") {
190
+ await capture("database_opened", {
191
+ mode,
192
+ has_custom_path: dbPath !== ":memory:",
193
+ });
194
+ }
195
+ // Vector search and batch insert tracking removed - only database_opened is tracked
196
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics.js","sourceRoot":"","sources":["../../src/analytics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SochDB Analytics - Anonymous usage tracking with PostHog\n *\n * This module provides anonymous, privacy-respecting analytics to help\n * improve SochDB. All tracking can be disabled by setting:\n *\n *     SOCHDB_DISABLE_ANALYTICS=true\n *\n * No personally identifiable information (PII) is collected. Only aggregate\n * usage patterns are tracked to understand:\n * - Which features are most used\n * - Performance characteristics\n * - Error patterns for debugging\n *\n * Copyright 2025 Sushanth (https://github.com/sushanthpy)\n * Licensed under the Apache License, Version 2.0\n */\n\nimport { createHash } from \"crypto\";\nimport { hostname, platform, arch, release } from \"os\";\n\n// PostHog configuration\nconst POSTHOG_API_KEY = \"phc_zf0hm6ZmPUJj1pM07Kigqvphh1ClhKX1NahRU4G0bfu\";\nconst POSTHOG_HOST = \"https://us.i.posthog.com\";\n\n// Lazy-loaded PostHog client\nlet posthogClient: any = null;\nlet posthogInitialized = false;\n\n/**\n * Check if analytics is disabled via environment variable.\n * \n * Analytics is disabled when SOCHDB_DISABLE_ANALYTICS is set to 'true', '1', 'yes', or 'on'.\n * \n * @returns true if analytics is disabled\n */\nexport function isAnalyticsDisabled(): boolean {\n  const disableVar = (\n    process.env.SOCHDB_DISABLE_ANALYTICS || \"\"\n  ).toLowerCase();\n  return [\"true\", \"1\", \"yes\", \"on\"].includes(disableVar);\n}\n\n/**\n * Generate a stable anonymous ID for this machine.\n *\n * Uses a hash of machine-specific but non-identifying information.\n * The same machine will always get the same ID, but the ID cannot\n * be reversed to identify the machine.\n */\nfunction getAnonymousId(): string {\n  try {\n    const machineInfo = [\n      hostname(),\n      platform(),\n      arch(),\n      process.getuid?.() ?? \"windows\",\n    ].join(\"|\");\n\n    return createHash(\"sha256\").update(machineInfo).digest(\"hex\").slice(0, 16);\n  } catch {\n    return \"anonymous\";\n  }\n}\n\n/**\n * Lazily initialize PostHog client.\n */\nasync function getPosthogClient(): Promise<any> {\n  if (isAnalyticsDisabled()) {\n    return null;\n  }\n\n  if (posthogInitialized) {\n    return posthogClient;\n  }\n\n  posthogInitialized = true;\n\n  try {\n    // Use require to avoid TypeScript checking the optional module\n    // This allows the SDK to work without posthog-node installed\n    // eslint-disable-next-line @typescript-eslint/no-var-requires\n    const posthogModule = require(\"posthog-node\");\n    const { PostHog } = posthogModule;\n    posthogClient = new PostHog(POSTHOG_API_KEY, {\n      host: POSTHOG_HOST,\n    });\n    return posthogClient;\n  } catch {\n    // posthog-node not installed or error - analytics disabled\n    return null;\n  }\n}\n\n/**\n * Get the SDK version.\n */\nfunction getSdkVersion(): string {\n  try {\n    // eslint-disable-next-line @typescript-eslint/no-var-requires\n    const pkg = require(\"../package.json\");\n    return pkg.version || \"unknown\";\n  } catch {\n    return \"unknown\";\n  }\n}\n\nexport interface EventProperties {\n  [key: string]: string | number | boolean | null | undefined;\n}\n\n/**\n * Capture an analytics event.\n *\n * This function is a no-op if:\n * - SOCHDB_DISABLE_ANALYTICS=true\n * - posthog-node package is not installed\n * - Any error occurs (fails silently)\n *\n * @param event - Event name (e.g., \"database_opened\", \"vector_search\")\n * @param properties - Optional event properties\n * @param distinctId - Optional distinct ID (defaults to anonymous machine ID)\n */\nexport async function capture(\n  event: string,\n  properties?: EventProperties,\n  distinctId?: string\n): Promise<void> {\n  if (isAnalyticsDisabled()) {\n    return;\n  }\n\n  try {\n    const client = await getPosthogClient();\n    if (!client) {\n      return;\n    }\n\n    // Build properties with SDK context\n    const eventProperties: EventProperties = {\n      sdk: \"nodejs\",\n      sdk_version: getSdkVersion(),\n      node_version: process.version,\n      os: platform(),\n      arch: arch(),\n      ...properties,\n    };\n\n    client.capture({\n      distinctId: distinctId || getAnonymousId(),\n      event,\n      properties: eventProperties,\n    });\n    \n    // Flush to ensure event is sent immediately\n    await client.flush();\n  } catch {\n    // Never let analytics break user code\n  }\n}\n\n/**\n * Capture an error event for debugging.\n *\n * Only sends static information - no dynamic error messages.\n *\n * @param errorType - Static error category (e.g., \"connection_error\", \"query_error\", \"timeout_error\")\n * @param location - Static code location (e.g., \"database.open\", \"query.execute\", \"transaction.commit\")\n * @param properties - Optional additional static properties only\n *\n * @example\n * ```typescript\n * captureError('connection_error', 'database.open');\n * captureError('query_error', 'sql.execute', { query_type: 'SELECT' });\n * ```\n */\nexport async function captureError(\n  errorType: string,\n  location: string,\n  properties?: EventProperties\n): Promise<void> {\n  await capture(\"error\", {\n    error_type: errorType,\n    location,\n    ...properties,\n  });\n}\n\n/**\n * Flush any pending events and shutdown the client.\n */\nexport async function shutdown(): Promise<void> {\n  if (isAnalyticsDisabled()) {\n    return;\n  }\n\n  try {\n    if (posthogClient) {\n      await posthogClient.shutdown();\n    }\n  } catch {\n    // Ignore shutdown errors\n  }\n}\n\n// Convenience functions for common events\n\n/**\n * Track database open event.\n */\nexport async function trackDatabaseOpen(\n  dbPath: string,\n  mode: string = \"embedded\"\n): Promise<void> {\n  await capture(\"database_opened\", {\n    mode,\n    has_custom_path: dbPath !== \":memory:\",\n  });\n}\n\n// Vector search and batch insert tracking removed - only database_opened is tracked\n"]}