@friggframework/core 2.0.0-next.6 → 2.0.0-next.60
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/CLAUDE.md +694 -0
- package/README.md +959 -50
- package/application/commands/README.md +451 -0
- package/application/commands/credential-commands.js +245 -0
- package/application/commands/entity-commands.js +336 -0
- package/application/commands/integration-commands.js +210 -0
- package/application/commands/user-commands.js +283 -0
- package/application/index.js +69 -0
- package/core/CLAUDE.md +690 -0
- package/core/Worker.js +8 -21
- package/core/create-handler.js +14 -7
- package/credential/repositories/credential-repository-documentdb.js +304 -0
- package/credential/repositories/credential-repository-factory.js +54 -0
- package/credential/repositories/credential-repository-interface.js +98 -0
- package/credential/repositories/credential-repository-mongo.js +269 -0
- package/credential/repositories/credential-repository-postgres.js +291 -0
- package/credential/repositories/credential-repository.js +302 -0
- package/credential/use-cases/get-credential-for-user.js +25 -0
- package/credential/use-cases/update-authentication-status.js +15 -0
- package/database/MONGODB_TRANSACTION_FIX.md +198 -0
- package/database/adapters/lambda-invoker.js +97 -0
- package/database/config.js +154 -0
- package/database/documentdb-encryption-service.js +330 -0
- package/database/documentdb-utils.js +136 -0
- package/database/encryption/README.md +839 -0
- package/database/encryption/documentdb-encryption-service.md +3575 -0
- package/database/encryption/encryption-schema-registry.js +268 -0
- package/database/encryption/field-encryption-service.js +226 -0
- package/database/encryption/logger.js +79 -0
- package/database/encryption/prisma-encryption-extension.js +222 -0
- package/database/index.js +61 -21
- package/database/models/WebsocketConnection.js +16 -10
- package/database/models/readme.md +1 -0
- package/database/prisma.js +182 -0
- package/database/repositories/health-check-repository-documentdb.js +134 -0
- package/database/repositories/health-check-repository-factory.js +48 -0
- package/database/repositories/health-check-repository-interface.js +82 -0
- package/database/repositories/health-check-repository-mongodb.js +89 -0
- package/database/repositories/health-check-repository-postgres.js +82 -0
- package/database/repositories/health-check-repository.js +108 -0
- package/database/repositories/migration-status-repository-s3.js +137 -0
- package/database/use-cases/check-database-health-use-case.js +29 -0
- package/database/use-cases/check-database-state-use-case.js +81 -0
- package/database/use-cases/check-encryption-health-use-case.js +83 -0
- package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
- package/database/use-cases/get-migration-status-use-case.js +93 -0
- package/database/use-cases/run-database-migration-use-case.js +139 -0
- package/database/use-cases/test-encryption-use-case.js +253 -0
- package/database/use-cases/trigger-database-migration-use-case.js +157 -0
- package/database/utils/mongodb-collection-utils.js +91 -0
- package/database/utils/mongodb-schema-init.js +106 -0
- package/database/utils/prisma-runner.js +477 -0
- package/database/utils/prisma-schema-parser.js +182 -0
- package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
- package/encrypt/Cryptor.js +34 -168
- package/encrypt/index.js +1 -2
- package/encrypt/test-encrypt.js +0 -2
- package/errors/client-safe-error.js +26 -0
- package/errors/fetch-error.js +2 -1
- package/errors/index.js +2 -0
- package/generated/prisma-mongodb/client.d.ts +1 -0
- package/generated/prisma-mongodb/client.js +4 -0
- package/generated/prisma-mongodb/default.d.ts +1 -0
- package/generated/prisma-mongodb/default.js +4 -0
- package/generated/prisma-mongodb/edge.d.ts +1 -0
- package/generated/prisma-mongodb/edge.js +334 -0
- package/generated/prisma-mongodb/index-browser.js +316 -0
- package/generated/prisma-mongodb/index.d.ts +22903 -0
- package/generated/prisma-mongodb/index.js +359 -0
- package/generated/prisma-mongodb/package.json +183 -0
- package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
- package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
- package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
- package/generated/prisma-mongodb/runtime/binary.js +289 -0
- package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
- package/generated/prisma-mongodb/runtime/edge.js +34 -0
- package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
- package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
- package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
- package/generated/prisma-mongodb/runtime/react-native.js +83 -0
- package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
- package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
- package/generated/prisma-mongodb/schema.prisma +360 -0
- package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
- package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
- package/generated/prisma-mongodb/wasm.d.ts +1 -0
- package/generated/prisma-mongodb/wasm.js +341 -0
- package/generated/prisma-postgresql/client.d.ts +1 -0
- package/generated/prisma-postgresql/client.js +4 -0
- package/generated/prisma-postgresql/default.d.ts +1 -0
- package/generated/prisma-postgresql/default.js +4 -0
- package/generated/prisma-postgresql/edge.d.ts +1 -0
- package/generated/prisma-postgresql/edge.js +356 -0
- package/generated/prisma-postgresql/index-browser.js +338 -0
- package/generated/prisma-postgresql/index.d.ts +25077 -0
- package/generated/prisma-postgresql/index.js +381 -0
- package/generated/prisma-postgresql/package.json +183 -0
- package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
- package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
- package/generated/prisma-postgresql/query_engine_bg.js +2 -0
- package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
- package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
- package/generated/prisma-postgresql/runtime/binary.js +289 -0
- package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
- package/generated/prisma-postgresql/runtime/edge.js +34 -0
- package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
- package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
- package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
- package/generated/prisma-postgresql/runtime/react-native.js +83 -0
- package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
- package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
- package/generated/prisma-postgresql/schema.prisma +343 -0
- package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
- package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
- package/generated/prisma-postgresql/wasm.d.ts +1 -0
- package/generated/prisma-postgresql/wasm.js +363 -0
- package/handlers/WEBHOOKS.md +653 -0
- package/handlers/app-definition-loader.js +38 -0
- package/handlers/app-handler-helpers.js +56 -0
- package/handlers/backend-utils.js +186 -0
- package/handlers/database-migration-handler.js +227 -0
- package/handlers/integration-event-dispatcher.js +54 -0
- package/handlers/routers/HEALTHCHECK.md +342 -0
- package/handlers/routers/auth.js +15 -0
- package/handlers/routers/db-migration.handler.js +29 -0
- package/handlers/routers/db-migration.js +326 -0
- package/handlers/routers/health.js +516 -0
- package/handlers/routers/integration-defined-routers.js +45 -0
- package/handlers/routers/integration-webhook-routers.js +67 -0
- package/handlers/routers/user.js +63 -0
- package/handlers/routers/websocket.js +57 -0
- package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
- package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
- package/handlers/workers/db-migration.js +352 -0
- package/handlers/workers/integration-defined-workers.js +27 -0
- package/index.js +77 -22
- package/integrations/WEBHOOK-QUICKSTART.md +151 -0
- package/integrations/index.js +12 -10
- package/integrations/integration-base.js +326 -55
- package/integrations/integration-router.js +374 -179
- package/integrations/options.js +1 -1
- package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
- package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
- package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
- package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
- package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
- package/integrations/repositories/integration-mapping-repository.js +156 -0
- package/integrations/repositories/integration-repository-documentdb.js +210 -0
- package/integrations/repositories/integration-repository-factory.js +51 -0
- package/integrations/repositories/integration-repository-interface.js +127 -0
- package/integrations/repositories/integration-repository-mongo.js +303 -0
- package/integrations/repositories/integration-repository-postgres.js +352 -0
- package/integrations/repositories/process-repository-documentdb.js +243 -0
- package/integrations/repositories/process-repository-factory.js +53 -0
- package/integrations/repositories/process-repository-interface.js +90 -0
- package/integrations/repositories/process-repository-mongo.js +190 -0
- package/integrations/repositories/process-repository-postgres.js +217 -0
- package/integrations/tests/doubles/dummy-integration-class.js +83 -0
- package/integrations/tests/doubles/test-integration-repository.js +99 -0
- package/integrations/use-cases/create-integration.js +83 -0
- package/integrations/use-cases/create-process.js +128 -0
- package/integrations/use-cases/delete-integration-for-user.js +101 -0
- package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
- package/integrations/use-cases/get-integration-for-user.js +78 -0
- package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
- package/integrations/use-cases/get-integration-instance.js +83 -0
- package/integrations/use-cases/get-integrations-for-user.js +88 -0
- package/integrations/use-cases/get-possible-integrations.js +27 -0
- package/integrations/use-cases/get-process.js +87 -0
- package/integrations/use-cases/index.js +19 -0
- package/integrations/use-cases/load-integration-context.js +71 -0
- package/integrations/use-cases/update-integration-messages.js +44 -0
- package/integrations/use-cases/update-integration-status.js +32 -0
- package/integrations/use-cases/update-integration.js +93 -0
- package/integrations/use-cases/update-process-metrics.js +201 -0
- package/integrations/use-cases/update-process-state.js +119 -0
- package/integrations/utils/map-integration-dto.js +37 -0
- package/jest-global-setup-noop.js +3 -0
- package/jest-global-teardown-noop.js +3 -0
- package/logs/logger.js +0 -4
- package/{module-plugin → modules}/entity.js +1 -1
- package/{module-plugin → modules}/index.js +0 -8
- package/modules/module-factory.js +56 -0
- package/modules/module.js +221 -0
- package/modules/repositories/module-repository-documentdb.js +307 -0
- package/modules/repositories/module-repository-factory.js +40 -0
- package/modules/repositories/module-repository-interface.js +129 -0
- package/modules/repositories/module-repository-mongo.js +377 -0
- package/modules/repositories/module-repository-postgres.js +426 -0
- package/modules/repositories/module-repository.js +316 -0
- package/modules/requester/api-key.js +52 -0
- package/{module-plugin → modules}/requester/requester.js +1 -0
- package/{module-plugin → modules}/test/mock-api/api.js +8 -3
- package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
- package/modules/tests/doubles/test-module-factory.js +16 -0
- package/modules/tests/doubles/test-module-repository.js +39 -0
- package/modules/use-cases/get-entities-for-user.js +32 -0
- package/modules/use-cases/get-entity-options-by-id.js +71 -0
- package/modules/use-cases/get-entity-options-by-type.js +34 -0
- package/modules/use-cases/get-module-instance-from-type.js +31 -0
- package/modules/use-cases/get-module.js +74 -0
- package/modules/use-cases/process-authorization-callback.js +133 -0
- package/modules/use-cases/refresh-entity-options.js +72 -0
- package/modules/use-cases/test-module-auth.js +72 -0
- package/modules/utils/map-module-dto.js +18 -0
- package/package.json +82 -50
- package/prisma-mongodb/schema.prisma +360 -0
- package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
- package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
- package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
- package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
- package/prisma-postgresql/migrations/migration_lock.toml +3 -0
- package/prisma-postgresql/schema.prisma +343 -0
- package/queues/queuer-util.js +27 -22
- package/syncs/manager.js +468 -443
- package/syncs/repositories/sync-repository-documentdb.js +240 -0
- package/syncs/repositories/sync-repository-factory.js +43 -0
- package/syncs/repositories/sync-repository-interface.js +109 -0
- package/syncs/repositories/sync-repository-mongo.js +239 -0
- package/syncs/repositories/sync-repository-postgres.js +319 -0
- package/syncs/sync.js +0 -1
- package/token/repositories/token-repository-documentdb.js +137 -0
- package/token/repositories/token-repository-factory.js +40 -0
- package/token/repositories/token-repository-interface.js +131 -0
- package/token/repositories/token-repository-mongo.js +219 -0
- package/token/repositories/token-repository-postgres.js +264 -0
- package/token/repositories/token-repository.js +219 -0
- package/types/core/index.d.ts +2 -2
- package/types/integrations/index.d.ts +2 -6
- package/types/module-plugin/index.d.ts +5 -59
- package/types/syncs/index.d.ts +0 -2
- package/user/repositories/user-repository-documentdb.js +441 -0
- package/user/repositories/user-repository-factory.js +52 -0
- package/user/repositories/user-repository-interface.js +201 -0
- package/user/repositories/user-repository-mongo.js +308 -0
- package/user/repositories/user-repository-postgres.js +360 -0
- package/user/tests/doubles/test-user-repository.js +72 -0
- package/user/use-cases/authenticate-user.js +127 -0
- package/user/use-cases/authenticate-with-shared-secret.js +48 -0
- package/user/use-cases/create-individual-user.js +61 -0
- package/user/use-cases/create-organization-user.js +47 -0
- package/user/use-cases/create-token-for-user-id.js +30 -0
- package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
- package/user/use-cases/get-user-from-bearer-token.js +77 -0
- package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
- package/user/use-cases/login-user.js +122 -0
- package/user/user.js +125 -0
- package/utils/backend-path.js +38 -0
- package/utils/index.js +6 -0
- package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
- package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
- package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
- package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
- package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
- package/websocket/repositories/websocket-connection-repository.js +161 -0
- package/database/models/State.js +0 -9
- package/database/models/Token.js +0 -70
- package/database/mongo.js +0 -45
- package/encrypt/Cryptor.test.js +0 -32
- package/encrypt/encrypt.js +0 -132
- package/encrypt/encrypt.test.js +0 -1069
- package/errors/base-error.test.js +0 -32
- package/errors/fetch-error.test.js +0 -79
- package/errors/halt-error.test.js +0 -11
- package/errors/validation-errors.test.js +0 -120
- package/integrations/create-frigg-backend.js +0 -31
- package/integrations/integration-factory.js +0 -251
- package/integrations/integration-mapping.js +0 -43
- package/integrations/integration-model.js +0 -46
- package/integrations/integration-user.js +0 -144
- package/integrations/test/integration-base.test.js +0 -144
- package/lambda/TimeoutCatcher.test.js +0 -68
- package/logs/logger.test.js +0 -76
- package/module-plugin/auther.js +0 -393
- package/module-plugin/credential.js +0 -22
- package/module-plugin/entity-manager.js +0 -70
- package/module-plugin/manager.js +0 -169
- package/module-plugin/module-factory.js +0 -61
- package/module-plugin/requester/api-key.js +0 -36
- package/module-plugin/requester/requester.test.js +0 -28
- package/module-plugin/test/auther.test.js +0 -97
- /package/{module-plugin → modules}/ModuleConstants.js +0 -0
- /package/{module-plugin → modules}/requester/basic.js +0 -0
- /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
- /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Database Configuration
|
|
3
|
+
* Manages configuration for Prisma ORM operations
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Determines database type from environment or app definition
|
|
8
|
+
*
|
|
9
|
+
* Detection order:
|
|
10
|
+
* 1. DB_TYPE environment variable (set for migration handlers)
|
|
11
|
+
* 2. App definition (backend/index.js Definition.database configuration)
|
|
12
|
+
*
|
|
13
|
+
* @returns {'mongodb'|'postgresql'|'documentdb'} Database type
|
|
14
|
+
* @throws {Error} If database type cannot be determined or app definition missing
|
|
15
|
+
*/
|
|
16
|
+
function getDatabaseType() {
|
|
17
|
+
// First, check DB_TYPE environment variable (migration handlers set this)
|
|
18
|
+
if (process.env.DB_TYPE) {
|
|
19
|
+
return process.env.DB_TYPE;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
// Fallback: Load app definition
|
|
23
|
+
try {
|
|
24
|
+
const path = require('node:path');
|
|
25
|
+
const fs = require('node:fs');
|
|
26
|
+
const { findNearestBackendPackageJson } = require('../utils');
|
|
27
|
+
|
|
28
|
+
let backendIndexPath;
|
|
29
|
+
let database;
|
|
30
|
+
const backendPackagePath = findNearestBackendPackageJson();
|
|
31
|
+
|
|
32
|
+
if (!backendPackagePath) {
|
|
33
|
+
throw new Error(
|
|
34
|
+
'[Frigg] Cannot find backend package.json. ' +
|
|
35
|
+
'Ensure backend/package.json exists in your project.'
|
|
36
|
+
);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
const backendDir = path.dirname(backendPackagePath);
|
|
40
|
+
backendIndexPath = path.join(backendDir, 'index.js');
|
|
41
|
+
|
|
42
|
+
if (!fs.existsSync(backendIndexPath)) {
|
|
43
|
+
throw new Error(
|
|
44
|
+
`[Frigg] Backend index.js not found at ${backendIndexPath}. ` +
|
|
45
|
+
'Ensure backend/index.js exists with a Definition export.'
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
let backendModule;
|
|
50
|
+
try {
|
|
51
|
+
backendModule = require(backendIndexPath);
|
|
52
|
+
} catch (requireError) {
|
|
53
|
+
// Extract the actual file with the error from the stack trace
|
|
54
|
+
// Skip internal Node.js files (node:internal/*) and find first user file
|
|
55
|
+
let errorFile = 'unknown file';
|
|
56
|
+
const stackLines = requireError.stack?.split('\n') || [];
|
|
57
|
+
|
|
58
|
+
for (const line of stackLines) {
|
|
59
|
+
// Match file paths in stack trace, excluding node:internal
|
|
60
|
+
const match = line.match(/\(([^)]+\.js):\d+:\d+\)/) || line.match(/at ([^(]+\.js):\d+:\d+/);
|
|
61
|
+
if (match && match[1] && !match[1].includes('node:internal')) {
|
|
62
|
+
errorFile = match[1];
|
|
63
|
+
break;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Provide better error context for syntax/runtime errors
|
|
68
|
+
throw new Error(
|
|
69
|
+
`[Frigg] Failed to load app definition from ${backendIndexPath}\n` +
|
|
70
|
+
`Error: ${requireError.message}\n` +
|
|
71
|
+
`File with error: ${errorFile}\n` +
|
|
72
|
+
`\nFull stack trace:\n${requireError.stack}\n\n` +
|
|
73
|
+
'This error occurred while loading your app definition or its dependencies. ' +
|
|
74
|
+
'Check the file listed above for syntax errors (trailing commas, missing brackets, etc.)'
|
|
75
|
+
);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
database = backendModule?.Definition?.database;
|
|
79
|
+
|
|
80
|
+
if (!database) {
|
|
81
|
+
throw new Error(
|
|
82
|
+
'[Frigg] App definition missing database configuration. ' +
|
|
83
|
+
`Add database: { postgres: { enable: true } } (or mongoDB/documentDB) to ${backendIndexPath}`
|
|
84
|
+
);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// Determine database type from enabled database
|
|
88
|
+
// Priority order: postgres > mongoDB > documentDB
|
|
89
|
+
if (database.postgres?.enable === true) {
|
|
90
|
+
return 'postgresql';
|
|
91
|
+
}
|
|
92
|
+
if (database.mongoDB?.enable === true) {
|
|
93
|
+
return 'mongodb';
|
|
94
|
+
}
|
|
95
|
+
if (database.documentDB?.enable === true) {
|
|
96
|
+
return 'documentdb';
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
throw new Error(
|
|
100
|
+
'[Frigg] No database enabled in app definition. ' +
|
|
101
|
+
'Set one of: database.postgres.enable, database.mongoDB.enable, or database.documentDB.enable to true'
|
|
102
|
+
);
|
|
103
|
+
} catch (error) {
|
|
104
|
+
// Re-throw with context if it's our error
|
|
105
|
+
if (error.message.includes('[Frigg]')) {
|
|
106
|
+
throw error;
|
|
107
|
+
}
|
|
108
|
+
// Wrap unexpected errors
|
|
109
|
+
throw new Error(
|
|
110
|
+
`[Frigg] Failed to determine database type: ${error.message}`
|
|
111
|
+
);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Cached database type (lazy evaluation)
|
|
117
|
+
* @type {'mongodb'|'postgresql'|'documentdb'|null}
|
|
118
|
+
*/
|
|
119
|
+
let cachedDbType = null;
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Enable Prisma debug logging
|
|
123
|
+
* Set PRISMA_LOG_LEVEL to comma-separated list: query,info,warn,error
|
|
124
|
+
* @type {string}
|
|
125
|
+
*/
|
|
126
|
+
const PRISMA_LOG_LEVEL = process.env.PRISMA_LOG_LEVEL || 'error,warn';
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* Enable Prisma query logging for performance monitoring
|
|
130
|
+
* @type {boolean}
|
|
131
|
+
*/
|
|
132
|
+
const PRISMA_QUERY_LOGGING = process.env.PRISMA_QUERY_LOGGING === 'true';
|
|
133
|
+
|
|
134
|
+
module.exports = {
|
|
135
|
+
getDatabaseType, // Export for testing and direct use
|
|
136
|
+
PRISMA_LOG_LEVEL,
|
|
137
|
+
PRISMA_QUERY_LOGGING,
|
|
138
|
+
};
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* Lazy-evaluated database type determined from app definition
|
|
142
|
+
* Only evaluates when accessed, preventing module load failures in test environments
|
|
143
|
+
* @type {'mongodb'|'postgresql'|'documentdb'}
|
|
144
|
+
*/
|
|
145
|
+
Object.defineProperty(module.exports, 'DB_TYPE', {
|
|
146
|
+
get() {
|
|
147
|
+
if (cachedDbType === null) {
|
|
148
|
+
cachedDbType = getDatabaseType();
|
|
149
|
+
}
|
|
150
|
+
return cachedDbType;
|
|
151
|
+
},
|
|
152
|
+
enumerable: true,
|
|
153
|
+
configurable: true
|
|
154
|
+
});
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
const { Cryptor } = require('../encrypt/Cryptor');
|
|
2
|
+
const { getEncryptedFields, loadCustomEncryptionSchema } = require('./encryption/encryption-schema-registry');
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Encryption service specifically for DocumentDB repositories
|
|
6
|
+
* that use $runCommandRaw and bypass Prisma Client Extensions.
|
|
7
|
+
*
|
|
8
|
+
* Provides document-level encryption/decryption, handling nested fields
|
|
9
|
+
* according to the encryption schema registry.
|
|
10
|
+
*
|
|
11
|
+
* @class DocumentDBEncryptionService
|
|
12
|
+
* @example
|
|
13
|
+
* const service = new DocumentDBEncryptionService();
|
|
14
|
+
*
|
|
15
|
+
* // Encrypt before write
|
|
16
|
+
* const encrypted = await service.encryptFields('Credential', document);
|
|
17
|
+
* await insertOne(prisma, 'Credential', encrypted);
|
|
18
|
+
*
|
|
19
|
+
* // Decrypt after read
|
|
20
|
+
* const doc = await findOne(prisma, 'Credential', filter);
|
|
21
|
+
* const decrypted = await service.decryptFields('Credential', doc);
|
|
22
|
+
*/
|
|
23
|
+
class DocumentDBEncryptionService {
|
|
24
|
+
/**
|
|
25
|
+
* @param {Object} options - Configuration options
|
|
26
|
+
* @param {Cryptor} [options.cryptor] - Optional Cryptor instance for dependency injection (useful for testing)
|
|
27
|
+
*/
|
|
28
|
+
constructor({ cryptor = null } = {}) {
|
|
29
|
+
if (cryptor) {
|
|
30
|
+
// Dependency injection - use provided Cryptor (for testing)
|
|
31
|
+
this.cryptor = cryptor;
|
|
32
|
+
this.enabled = true;
|
|
33
|
+
} else {
|
|
34
|
+
// Default behavior - create Cryptor from environment
|
|
35
|
+
this._initializeCryptor();
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Initialize Cryptor with environment-based configuration.
|
|
41
|
+
* Matches the logic from @friggframework/core/database/prisma.js
|
|
42
|
+
*
|
|
43
|
+
* Encryption is bypassed in dev/test/local stages.
|
|
44
|
+
* Production uses AWS KMS (if available) or AES encryption.
|
|
45
|
+
*
|
|
46
|
+
* @private
|
|
47
|
+
*/
|
|
48
|
+
_initializeCryptor() {
|
|
49
|
+
// Load custom encryption schema from app definition BEFORE checking configuration
|
|
50
|
+
// This ensures custom fields (like User.username) are registered before any encryption operations
|
|
51
|
+
loadCustomEncryptionSchema();
|
|
52
|
+
|
|
53
|
+
// Match logic from packages/core/database/prisma.js
|
|
54
|
+
const stage = process.env.STAGE || process.env.NODE_ENV || 'development';
|
|
55
|
+
const bypassEncryption = ['dev', 'test', 'local'].includes(stage.toLowerCase());
|
|
56
|
+
|
|
57
|
+
if (bypassEncryption) {
|
|
58
|
+
this.cryptor = null;
|
|
59
|
+
this.enabled = false;
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// Determine encryption method (ensure boolean values)
|
|
64
|
+
const hasKMS = !!(process.env.KMS_KEY_ARN && process.env.KMS_KEY_ARN.trim() !== '');
|
|
65
|
+
const hasAES = !!(process.env.AES_KEY_ID && process.env.AES_KEY_ID.trim() !== '');
|
|
66
|
+
|
|
67
|
+
if (!hasKMS && !hasAES) {
|
|
68
|
+
console.warn('[DocumentDBEncryptionService] No encryption keys configured. Encryption disabled.');
|
|
69
|
+
this.cryptor = null;
|
|
70
|
+
this.enabled = false;
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// KMS takes precedence over AES
|
|
75
|
+
const shouldUseAws = hasKMS;
|
|
76
|
+
this.cryptor = new Cryptor({ shouldUseAws });
|
|
77
|
+
this.enabled = true;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* Encrypt sensitive fields in a document before storing to DocumentDB.
|
|
82
|
+
*
|
|
83
|
+
* Reads field paths from encryption-schema-registry.js and encrypts
|
|
84
|
+
* only the fields defined for the given model.
|
|
85
|
+
*
|
|
86
|
+
* @param {string} modelName - Model name from schema registry (e.g., 'User', 'Credential')
|
|
87
|
+
* @param {Object} document - Document to encrypt
|
|
88
|
+
* @returns {Promise<Object>} - New document with encrypted fields (original unchanged)
|
|
89
|
+
*
|
|
90
|
+
* @example
|
|
91
|
+
* const plainDoc = {
|
|
92
|
+
* userId: '123',
|
|
93
|
+
* data: { access_token: 'plain_secret' }
|
|
94
|
+
* };
|
|
95
|
+
* const encrypted = await service.encryptFields('Credential', plainDoc);
|
|
96
|
+
* // encrypted.data.access_token = "keyId:iv:cipher:encKey"
|
|
97
|
+
*/
|
|
98
|
+
async encryptFields(modelName, document) {
|
|
99
|
+
// Bypass if encryption disabled
|
|
100
|
+
if (!this.enabled || !this.cryptor) {
|
|
101
|
+
return document;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Validate input
|
|
105
|
+
if (!document || typeof document !== 'object') {
|
|
106
|
+
return document;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// Get encrypted fields from registry
|
|
110
|
+
const encryptedFieldsConfig = getEncryptedFields(modelName);
|
|
111
|
+
if (!encryptedFieldsConfig || encryptedFieldsConfig.length === 0) {
|
|
112
|
+
return document;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// Deep clone to prevent mutation (preserves Date, RegExp, Buffer)
|
|
116
|
+
const result = structuredClone(document);
|
|
117
|
+
|
|
118
|
+
// Encrypt each field path
|
|
119
|
+
for (const fieldPath of encryptedFieldsConfig) {
|
|
120
|
+
await this._encryptFieldPath(result, fieldPath, modelName);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
return result;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Decrypt sensitive fields in a document after reading from DocumentDB.
|
|
128
|
+
*
|
|
129
|
+
* Reads field paths from encryption-schema-registry.js and decrypts
|
|
130
|
+
* only the fields defined for the given model.
|
|
131
|
+
*
|
|
132
|
+
* @param {string} modelName - Model name from schema registry
|
|
133
|
+
* @param {Object} document - Document to decrypt
|
|
134
|
+
* @returns {Promise<Object>} - New document with decrypted fields (original unchanged)
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* const encryptedDoc = {
|
|
138
|
+
* userId: '123',
|
|
139
|
+
* data: { access_token: 'keyId:iv:cipher:encKey' }
|
|
140
|
+
* };
|
|
141
|
+
* const decrypted = await service.decryptFields('Credential', encryptedDoc);
|
|
142
|
+
* // decrypted.data.access_token = "plain_secret"
|
|
143
|
+
*/
|
|
144
|
+
async decryptFields(modelName, document) {
|
|
145
|
+
// Bypass if encryption disabled
|
|
146
|
+
if (!this.enabled || !this.cryptor) {
|
|
147
|
+
return document;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Validate input
|
|
151
|
+
if (!document || typeof document !== 'object') {
|
|
152
|
+
return document;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
// Get encrypted fields from registry
|
|
156
|
+
const encryptedFieldsConfig = getEncryptedFields(modelName);
|
|
157
|
+
if (!encryptedFieldsConfig || encryptedFieldsConfig.length === 0) {
|
|
158
|
+
return document;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// Deep clone to prevent mutation (preserves Date, RegExp, Buffer)
|
|
162
|
+
const result = structuredClone(document);
|
|
163
|
+
|
|
164
|
+
// Decrypt each field path
|
|
165
|
+
for (const fieldPath of encryptedFieldsConfig) {
|
|
166
|
+
await this._decryptFieldPath(result, fieldPath, modelName);
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
return result;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
/**
|
|
173
|
+
* Encrypt a specific field path in a document (handles nested fields).
|
|
174
|
+
*
|
|
175
|
+
* @private
|
|
176
|
+
* @param {Object} document - Document to modify (mutated in place)
|
|
177
|
+
* @param {string} fieldPath - Field path from schema registry (e.g., 'data.access_token')
|
|
178
|
+
* @param {string} modelName - For error logging context
|
|
179
|
+
*/
|
|
180
|
+
async _encryptFieldPath(document, fieldPath, modelName) {
|
|
181
|
+
// Parse field path
|
|
182
|
+
const parts = fieldPath.split('.');
|
|
183
|
+
|
|
184
|
+
// Navigate to parent object
|
|
185
|
+
let current = document;
|
|
186
|
+
for (let i = 0; i < parts.length - 1; i++) {
|
|
187
|
+
if (!current[parts[i]]) {
|
|
188
|
+
// Path doesn't exist, nothing to encrypt
|
|
189
|
+
return;
|
|
190
|
+
}
|
|
191
|
+
current = current[parts[i]];
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
// Get field name and value
|
|
195
|
+
const fieldName = parts[parts.length - 1];
|
|
196
|
+
const value = current[fieldName];
|
|
197
|
+
|
|
198
|
+
// Skip if already encrypted or empty
|
|
199
|
+
if (!value || this._isEncryptedValue(value)) {
|
|
200
|
+
return;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
try {
|
|
204
|
+
// Convert to string if needed
|
|
205
|
+
const stringValue = typeof value === 'string'
|
|
206
|
+
? value
|
|
207
|
+
: JSON.stringify(value);
|
|
208
|
+
|
|
209
|
+
// Encrypt using Cryptor
|
|
210
|
+
current[fieldName] = await this.cryptor.encrypt(stringValue);
|
|
211
|
+
} catch (error) {
|
|
212
|
+
console.error(`[DocumentDBEncryptionService] Failed to encrypt ${modelName}.${fieldPath}:`, error.message);
|
|
213
|
+
throw error;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
/**
|
|
218
|
+
* Decrypt a specific field path in a document (handles nested fields).
|
|
219
|
+
*
|
|
220
|
+
* @private
|
|
221
|
+
* @param {Object} document - Document to modify (mutated in place)
|
|
222
|
+
* @param {string} fieldPath - Field path from schema registry
|
|
223
|
+
* @param {string} modelName - For error logging context
|
|
224
|
+
*/
|
|
225
|
+
async _decryptFieldPath(document, fieldPath, modelName) {
|
|
226
|
+
// Parse field path
|
|
227
|
+
const parts = fieldPath.split('.');
|
|
228
|
+
|
|
229
|
+
// Navigate to parent object
|
|
230
|
+
let current = document;
|
|
231
|
+
for (let i = 0; i < parts.length - 1; i++) {
|
|
232
|
+
if (!current[parts[i]]) {
|
|
233
|
+
// Path doesn't exist, nothing to decrypt
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
236
|
+
current = current[parts[i]];
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
// Get field name and encrypted value
|
|
240
|
+
const fieldName = parts[parts.length - 1];
|
|
241
|
+
const encryptedValue = current[fieldName];
|
|
242
|
+
|
|
243
|
+
// Skip if not encrypted format
|
|
244
|
+
if (!encryptedValue || !this._isEncryptedValue(encryptedValue)) {
|
|
245
|
+
return;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
try {
|
|
249
|
+
// Decrypt using Cryptor
|
|
250
|
+
const decryptedString = await this.cryptor.decrypt(encryptedValue);
|
|
251
|
+
|
|
252
|
+
// Try to parse as JSON (for objects/arrays)
|
|
253
|
+
try {
|
|
254
|
+
current[fieldName] = JSON.parse(decryptedString);
|
|
255
|
+
} catch {
|
|
256
|
+
// Not JSON, return as string
|
|
257
|
+
current[fieldName] = decryptedString;
|
|
258
|
+
}
|
|
259
|
+
} catch (error) {
|
|
260
|
+
const errorContext = {
|
|
261
|
+
modelName,
|
|
262
|
+
fieldPath,
|
|
263
|
+
encryptedValuePrefix: encryptedValue.substring(0, 20),
|
|
264
|
+
errorMessage: error.message
|
|
265
|
+
};
|
|
266
|
+
|
|
267
|
+
console.error(
|
|
268
|
+
`[DocumentDBEncryptionService] Failed to decrypt ${modelName}.${fieldPath}:`,
|
|
269
|
+
JSON.stringify(errorContext)
|
|
270
|
+
);
|
|
271
|
+
|
|
272
|
+
// Throw error to fail fast - don't silently corrupt data
|
|
273
|
+
throw new Error(`Decryption failed for ${modelName}.${fieldPath}: ${error.message}`);
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Check if a value is in encrypted format.
|
|
279
|
+
*
|
|
280
|
+
* Encrypted format: "keyId:iv:cipher:encKey" (envelope encryption)
|
|
281
|
+
* All parts are base64-encoded strings.
|
|
282
|
+
*
|
|
283
|
+
* @private
|
|
284
|
+
* @param {any} value - Value to check
|
|
285
|
+
* @returns {boolean} - True if value is encrypted
|
|
286
|
+
*
|
|
287
|
+
* @example
|
|
288
|
+
* _isEncryptedValue("plain_text") // false
|
|
289
|
+
* _isEncryptedValue("YWVzLWtleS0x:TXlJVkhlcmU=:QWN0dWFsQ2lwaGVy:RW5jcnlwdGVk") // true
|
|
290
|
+
* _isEncryptedValue(null) // false
|
|
291
|
+
* _isEncryptedValue({}) // false
|
|
292
|
+
*/
|
|
293
|
+
_isEncryptedValue(value) {
|
|
294
|
+
// Must be string
|
|
295
|
+
if (typeof value !== 'string') {
|
|
296
|
+
return false;
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
// Must have exactly 4 colon-separated parts
|
|
300
|
+
const parts = value.split(':');
|
|
301
|
+
if (parts.length !== 4) {
|
|
302
|
+
return false;
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
// Enhanced validation: check for base64 pattern
|
|
306
|
+
// This prevents false positives on URLs, connection strings, etc.
|
|
307
|
+
const base64Pattern = /^[A-Za-z0-9+/=]+$/;
|
|
308
|
+
|
|
309
|
+
// All parts should be base64-encoded
|
|
310
|
+
if (!parts.every(part => base64Pattern.test(part))) {
|
|
311
|
+
return false;
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
// Encrypted values should be sufficiently long to be valid
|
|
315
|
+
// Real encrypted values from Cryptor are always >50 chars due to envelope encryption format:
|
|
316
|
+
// - keyId (base64): ~12 chars minimum
|
|
317
|
+
// - iv (base64): ~24 chars for 16-byte IV
|
|
318
|
+
// - ciphertext (base64): varies, minimum ~16 chars for small values
|
|
319
|
+
// - encryptedKey (base64): ~44 chars for 32-byte data key
|
|
320
|
+
// Total minimum: ~96 chars, so 50 is a safe lower bound
|
|
321
|
+
// This prevents false positives on non-encrypted strings that happen to have 4 colons
|
|
322
|
+
if (value.length < 50) {
|
|
323
|
+
return false;
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
return true;
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
module.exports = { DocumentDBEncryptionService };
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
const { ObjectId } = require('mongodb');
|
|
2
|
+
|
|
3
|
+
function toObjectId(value) {
|
|
4
|
+
if (value === null || value === undefined || value === '') return undefined;
|
|
5
|
+
if (value instanceof ObjectId) return value;
|
|
6
|
+
if (typeof value === 'object' && value.$oid) return new ObjectId(value.$oid);
|
|
7
|
+
if (typeof value === 'string') return ObjectId.isValid(value) ? new ObjectId(value) : undefined;
|
|
8
|
+
return undefined;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
function toObjectIdArray(values) {
|
|
12
|
+
if (!Array.isArray(values)) return [];
|
|
13
|
+
return values.map(toObjectId).filter(Boolean);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function fromObjectId(value) {
|
|
17
|
+
if (value instanceof ObjectId) return value.toHexString();
|
|
18
|
+
if (typeof value === 'object' && value !== null && value.$oid) return value.$oid;
|
|
19
|
+
if (typeof value === 'string') return value;
|
|
20
|
+
return value === undefined || value === null ? value : String(value);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async function findMany(client, collection, filter = {}, options = {}) {
|
|
24
|
+
const command = { find: collection, filter };
|
|
25
|
+
if (options.projection) command.projection = options.projection;
|
|
26
|
+
if (options.sort) command.sort = options.sort;
|
|
27
|
+
if (options.limit) command.limit = options.limit;
|
|
28
|
+
const result = await client.$runCommandRaw(command);
|
|
29
|
+
return result?.cursor?.firstBatch || [];
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
async function findOne(client, collection, filter = {}, options = {}) {
|
|
33
|
+
const docs = await findMany(client, collection, filter, { ...options, limit: 1 });
|
|
34
|
+
return docs[0] || null;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
async function insertOne(client, collection, document) {
|
|
38
|
+
// Generate ObjectId if not present (MongoDB raw insert doesn't return insertedIds)
|
|
39
|
+
const _id = document._id || new ObjectId();
|
|
40
|
+
const docWithId = { ...document, _id };
|
|
41
|
+
|
|
42
|
+
const result = await client.$runCommandRaw({
|
|
43
|
+
insert: collection,
|
|
44
|
+
documents: [docWithId],
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
// Validate insert succeeded
|
|
48
|
+
if (result.ok !== 1) {
|
|
49
|
+
throw new Error(
|
|
50
|
+
`Insert command failed for collection '${collection}': ${JSON.stringify(result)}`
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// Check for write errors (duplicate keys, validation errors, etc.)
|
|
55
|
+
if (result.writeErrors && result.writeErrors.length > 0) {
|
|
56
|
+
const error = result.writeErrors[0];
|
|
57
|
+
const errorMsg = `Insert failed in '${collection}': ${error.errmsg} (code: ${error.code})`;
|
|
58
|
+
|
|
59
|
+
// Provide helpful context for common errors
|
|
60
|
+
if (error.code === 11000) {
|
|
61
|
+
throw new Error(`${errorMsg} - Duplicate key violation`);
|
|
62
|
+
}
|
|
63
|
+
throw new Error(errorMsg);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Verify exactly one document was inserted
|
|
67
|
+
if (result.n !== 1) {
|
|
68
|
+
throw new Error(
|
|
69
|
+
`Expected to insert 1 document into '${collection}', but inserted ${result.n}. ` +
|
|
70
|
+
`Result: ${JSON.stringify(result)}`
|
|
71
|
+
);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return _id;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
async function updateOne(client, collection, filter, update, options = {}) {
|
|
78
|
+
const updates = [{
|
|
79
|
+
q: filter,
|
|
80
|
+
u: update,
|
|
81
|
+
upsert: Boolean(options.upsert),
|
|
82
|
+
}];
|
|
83
|
+
if (options.arrayFilters) updates[0].arrayFilters = options.arrayFilters;
|
|
84
|
+
const result = await client.$runCommandRaw({
|
|
85
|
+
update: collection,
|
|
86
|
+
updates,
|
|
87
|
+
});
|
|
88
|
+
return result;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
async function deleteOne(client, collection, filter) {
|
|
92
|
+
return client.$runCommandRaw({
|
|
93
|
+
delete: collection,
|
|
94
|
+
deletes: [
|
|
95
|
+
{
|
|
96
|
+
q: filter,
|
|
97
|
+
limit: 1,
|
|
98
|
+
},
|
|
99
|
+
],
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
async function deleteMany(client, collection, filter) {
|
|
104
|
+
return client.$runCommandRaw({
|
|
105
|
+
delete: collection,
|
|
106
|
+
deletes: [
|
|
107
|
+
{
|
|
108
|
+
q: filter,
|
|
109
|
+
limit: 0,
|
|
110
|
+
},
|
|
111
|
+
],
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async function aggregate(client, collection, pipeline) {
|
|
116
|
+
const result = await client.$runCommandRaw({
|
|
117
|
+
aggregate: collection,
|
|
118
|
+
pipeline,
|
|
119
|
+
cursor: {},
|
|
120
|
+
});
|
|
121
|
+
return result?.cursor?.firstBatch || [];
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
module.exports = {
|
|
125
|
+
toObjectId,
|
|
126
|
+
toObjectIdArray,
|
|
127
|
+
fromObjectId,
|
|
128
|
+
findMany,
|
|
129
|
+
findOne,
|
|
130
|
+
insertOne,
|
|
131
|
+
updateOne,
|
|
132
|
+
deleteOne,
|
|
133
|
+
deleteMany,
|
|
134
|
+
aggregate,
|
|
135
|
+
};
|
|
136
|
+
|