@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.
- package/LICENSE +201 -0
- package/README.md +3349 -0
- package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
- package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
- package/bin/sochdb-bulk.js +80 -0
- package/bin/sochdb-grpc-server.js +80 -0
- package/bin/sochdb-server.js +84 -0
- package/dist/cjs/analytics.js +196 -0
- package/dist/cjs/database.js +929 -0
- package/dist/cjs/embedded/database.js +236 -0
- package/dist/cjs/embedded/ffi/bindings.js +113 -0
- package/dist/cjs/embedded/ffi/library-finder.js +135 -0
- package/dist/cjs/embedded/index.js +14 -0
- package/dist/cjs/embedded/transaction.js +172 -0
- package/dist/cjs/errors.js +71 -0
- package/dist/cjs/format.js +176 -0
- package/dist/cjs/grpc-client.js +328 -0
- package/dist/cjs/index.js +75 -0
- package/dist/cjs/ipc-client.js +504 -0
- package/dist/cjs/query.js +154 -0
- package/dist/cjs/server-manager.js +295 -0
- package/dist/cjs/sql-engine.js +874 -0
- package/dist/esm/analytics.js +196 -0
- package/dist/esm/database.js +931 -0
- package/dist/esm/embedded/database.js +239 -0
- package/dist/esm/embedded/ffi/bindings.js +142 -0
- package/dist/esm/embedded/ffi/library-finder.js +135 -0
- package/dist/esm/embedded/index.js +14 -0
- package/dist/esm/embedded/transaction.js +176 -0
- package/dist/esm/errors.js +71 -0
- package/dist/esm/format.js +179 -0
- package/dist/esm/grpc-client.js +333 -0
- package/dist/esm/index.js +75 -0
- package/dist/esm/ipc-client.js +505 -0
- package/dist/esm/query.js +159 -0
- package/dist/esm/server-manager.js +295 -0
- package/dist/esm/sql-engine.js +875 -0
- package/dist/types/analytics.d.ts +66 -0
- package/dist/types/analytics.d.ts.map +1 -0
- package/dist/types/database.d.ts +523 -0
- package/dist/types/database.d.ts.map +1 -0
- package/dist/types/embedded/database.d.ts +105 -0
- package/dist/types/embedded/database.d.ts.map +1 -0
- package/dist/types/embedded/ffi/bindings.d.ts +24 -0
- package/dist/types/embedded/ffi/bindings.d.ts.map +1 -0
- package/dist/types/embedded/ffi/library-finder.d.ts +17 -0
- package/dist/types/embedded/ffi/library-finder.d.ts.map +1 -0
- package/dist/types/embedded/index.d.ts +9 -0
- package/dist/types/embedded/index.d.ts.map +1 -0
- package/dist/types/embedded/transaction.d.ts +21 -0
- package/dist/types/embedded/transaction.d.ts.map +1 -0
- package/dist/types/errors.d.ts +36 -0
- package/dist/types/errors.d.ts.map +1 -0
- package/dist/types/format.d.ts +117 -0
- package/dist/types/format.d.ts.map +1 -0
- package/dist/types/grpc-client.d.ts +120 -0
- package/dist/types/grpc-client.d.ts.map +1 -0
- package/dist/types/index.d.ts +50 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/ipc-client.d.ts +177 -0
- package/dist/types/ipc-client.d.ts.map +1 -0
- package/dist/types/query.d.ts +85 -0
- package/dist/types/query.d.ts.map +1 -0
- package/dist/types/server-manager.d.ts +29 -0
- package/dist/types/server-manager.d.ts.map +1 -0
- package/dist/types/sql-engine.d.ts +100 -0
- package/dist/types/sql-engine.d.ts.map +1 -0
- package/package.json +90 -0
- package/scripts/postinstall.js +50 -0
|
@@ -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"]}
|