@friggframework/core 2.0.0-next.7 → 2.0.0-next.71

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 (293) hide show
  1. package/CLAUDE.md +694 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +451 -0
  4. package/application/commands/credential-commands.js +245 -0
  5. package/application/commands/entity-commands.js +336 -0
  6. package/application/commands/integration-commands.js +210 -0
  7. package/application/commands/scheduler-commands.js +263 -0
  8. package/application/commands/user-commands.js +283 -0
  9. package/application/index.js +73 -0
  10. package/core/CLAUDE.md +690 -0
  11. package/core/Worker.js +8 -21
  12. package/core/create-handler.js +14 -7
  13. package/credential/repositories/credential-repository-documentdb.js +304 -0
  14. package/credential/repositories/credential-repository-factory.js +54 -0
  15. package/credential/repositories/credential-repository-interface.js +98 -0
  16. package/credential/repositories/credential-repository-mongo.js +269 -0
  17. package/credential/repositories/credential-repository-postgres.js +287 -0
  18. package/credential/repositories/credential-repository.js +300 -0
  19. package/credential/use-cases/get-credential-for-user.js +25 -0
  20. package/credential/use-cases/update-authentication-status.js +15 -0
  21. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  22. package/database/adapters/lambda-invoker.js +97 -0
  23. package/database/config.js +154 -0
  24. package/database/documentdb-encryption-service.js +330 -0
  25. package/database/documentdb-utils.js +136 -0
  26. package/database/encryption/README.md +839 -0
  27. package/database/encryption/documentdb-encryption-service.md +3575 -0
  28. package/database/encryption/encryption-schema-registry.js +268 -0
  29. package/database/encryption/field-encryption-service.js +226 -0
  30. package/database/encryption/logger.js +79 -0
  31. package/database/encryption/prisma-encryption-extension.js +222 -0
  32. package/database/index.js +61 -21
  33. package/database/models/WebsocketConnection.js +16 -10
  34. package/database/models/readme.md +1 -0
  35. package/database/prisma.js +182 -0
  36. package/database/repositories/health-check-repository-documentdb.js +134 -0
  37. package/database/repositories/health-check-repository-factory.js +48 -0
  38. package/database/repositories/health-check-repository-interface.js +82 -0
  39. package/database/repositories/health-check-repository-mongodb.js +89 -0
  40. package/database/repositories/health-check-repository-postgres.js +82 -0
  41. package/database/repositories/health-check-repository.js +108 -0
  42. package/database/repositories/migration-status-repository-s3.js +137 -0
  43. package/database/use-cases/check-database-health-use-case.js +29 -0
  44. package/database/use-cases/check-database-state-use-case.js +81 -0
  45. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  46. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  47. package/database/use-cases/get-migration-status-use-case.js +93 -0
  48. package/database/use-cases/run-database-migration-use-case.js +139 -0
  49. package/database/use-cases/test-encryption-use-case.js +253 -0
  50. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  51. package/database/utils/mongodb-collection-utils.js +91 -0
  52. package/database/utils/mongodb-schema-init.js +106 -0
  53. package/database/utils/prisma-runner.js +477 -0
  54. package/database/utils/prisma-schema-parser.js +182 -0
  55. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  56. package/encrypt/Cryptor.js +34 -168
  57. package/encrypt/index.js +1 -2
  58. package/encrypt/test-encrypt.js +0 -2
  59. package/errors/client-safe-error.js +26 -0
  60. package/errors/fetch-error.js +2 -1
  61. package/errors/index.js +2 -0
  62. package/generated/prisma-mongodb/client.d.ts +1 -0
  63. package/generated/prisma-mongodb/client.js +4 -0
  64. package/generated/prisma-mongodb/default.d.ts +1 -0
  65. package/generated/prisma-mongodb/default.js +4 -0
  66. package/generated/prisma-mongodb/edge.d.ts +1 -0
  67. package/generated/prisma-mongodb/edge.js +335 -0
  68. package/generated/prisma-mongodb/index-browser.js +317 -0
  69. package/generated/prisma-mongodb/index.d.ts +22955 -0
  70. package/generated/prisma-mongodb/index.js +360 -0
  71. package/generated/prisma-mongodb/package.json +183 -0
  72. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  73. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  74. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  75. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  76. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  77. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  78. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  79. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  80. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  81. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  82. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  83. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  84. package/generated/prisma-mongodb/schema.prisma +362 -0
  85. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  86. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  87. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  88. package/generated/prisma-mongodb/wasm.js +342 -0
  89. package/generated/prisma-postgresql/client.d.ts +1 -0
  90. package/generated/prisma-postgresql/client.js +4 -0
  91. package/generated/prisma-postgresql/default.d.ts +1 -0
  92. package/generated/prisma-postgresql/default.js +4 -0
  93. package/generated/prisma-postgresql/edge.d.ts +1 -0
  94. package/generated/prisma-postgresql/edge.js +357 -0
  95. package/generated/prisma-postgresql/index-browser.js +339 -0
  96. package/generated/prisma-postgresql/index.d.ts +25131 -0
  97. package/generated/prisma-postgresql/index.js +382 -0
  98. package/generated/prisma-postgresql/package.json +183 -0
  99. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  100. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  101. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  102. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  103. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  104. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  105. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  106. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  107. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  108. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  109. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  110. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  111. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  112. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  113. package/generated/prisma-postgresql/schema.prisma +345 -0
  114. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  115. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  116. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  117. package/generated/prisma-postgresql/wasm.js +364 -0
  118. package/handlers/WEBHOOKS.md +653 -0
  119. package/handlers/app-definition-loader.js +38 -0
  120. package/handlers/app-handler-helpers.js +57 -0
  121. package/handlers/backend-utils.js +186 -0
  122. package/handlers/database-migration-handler.js +227 -0
  123. package/handlers/integration-event-dispatcher.js +54 -0
  124. package/handlers/routers/HEALTHCHECK.md +342 -0
  125. package/handlers/routers/auth.js +15 -0
  126. package/handlers/routers/db-migration.handler.js +29 -0
  127. package/handlers/routers/db-migration.js +326 -0
  128. package/handlers/routers/health.js +516 -0
  129. package/handlers/routers/integration-defined-routers.js +45 -0
  130. package/handlers/routers/integration-webhook-routers.js +67 -0
  131. package/handlers/routers/user.js +63 -0
  132. package/handlers/routers/websocket.js +57 -0
  133. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  134. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  135. package/handlers/workers/db-migration.js +352 -0
  136. package/handlers/workers/integration-defined-workers.js +27 -0
  137. package/index.js +78 -22
  138. package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
  139. package/infrastructure/scheduler/index.js +33 -0
  140. package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
  141. package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
  142. package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
  143. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  144. package/integrations/index.js +12 -10
  145. package/integrations/integration-base.js +326 -55
  146. package/integrations/integration-router.js +374 -179
  147. package/integrations/options.js +1 -1
  148. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  149. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  150. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  151. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  152. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  153. package/integrations/repositories/integration-mapping-repository.js +156 -0
  154. package/integrations/repositories/integration-repository-documentdb.js +210 -0
  155. package/integrations/repositories/integration-repository-factory.js +51 -0
  156. package/integrations/repositories/integration-repository-interface.js +127 -0
  157. package/integrations/repositories/integration-repository-mongo.js +303 -0
  158. package/integrations/repositories/integration-repository-postgres.js +352 -0
  159. package/integrations/repositories/process-repository-documentdb.js +243 -0
  160. package/integrations/repositories/process-repository-factory.js +53 -0
  161. package/integrations/repositories/process-repository-interface.js +90 -0
  162. package/integrations/repositories/process-repository-mongo.js +190 -0
  163. package/integrations/repositories/process-repository-postgres.js +217 -0
  164. package/integrations/tests/doubles/config-capturing-integration.js +81 -0
  165. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  166. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  167. package/integrations/use-cases/create-integration.js +83 -0
  168. package/integrations/use-cases/create-process.js +128 -0
  169. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  170. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  171. package/integrations/use-cases/get-integration-for-user.js +78 -0
  172. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  173. package/integrations/use-cases/get-integration-instance.js +83 -0
  174. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  175. package/integrations/use-cases/get-possible-integrations.js +27 -0
  176. package/integrations/use-cases/get-process.js +87 -0
  177. package/integrations/use-cases/index.js +19 -0
  178. package/integrations/use-cases/load-integration-context.js +71 -0
  179. package/integrations/use-cases/update-integration-messages.js +44 -0
  180. package/integrations/use-cases/update-integration-status.js +32 -0
  181. package/integrations/use-cases/update-integration.js +92 -0
  182. package/integrations/use-cases/update-process-metrics.js +201 -0
  183. package/integrations/use-cases/update-process-state.js +119 -0
  184. package/integrations/utils/map-integration-dto.js +37 -0
  185. package/jest-global-setup-noop.js +3 -0
  186. package/jest-global-teardown-noop.js +3 -0
  187. package/logs/logger.js +0 -4
  188. package/{module-plugin → modules}/entity.js +1 -1
  189. package/{module-plugin → modules}/index.js +0 -8
  190. package/modules/module-factory.js +56 -0
  191. package/modules/module.js +228 -0
  192. package/modules/repositories/module-repository-documentdb.js +335 -0
  193. package/modules/repositories/module-repository-factory.js +40 -0
  194. package/modules/repositories/module-repository-interface.js +129 -0
  195. package/modules/repositories/module-repository-mongo.js +408 -0
  196. package/modules/repositories/module-repository-postgres.js +453 -0
  197. package/modules/repositories/module-repository.js +345 -0
  198. package/modules/requester/api-key.js +52 -0
  199. package/modules/requester/oauth-2.js +396 -0
  200. package/{module-plugin → modules}/requester/requester.js +4 -2
  201. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  202. package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
  203. package/modules/tests/doubles/test-module-factory.js +16 -0
  204. package/modules/tests/doubles/test-module-repository.js +39 -0
  205. package/modules/use-cases/get-entities-for-user.js +32 -0
  206. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  207. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  208. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  209. package/modules/use-cases/get-module.js +74 -0
  210. package/modules/use-cases/process-authorization-callback.js +133 -0
  211. package/modules/use-cases/refresh-entity-options.js +72 -0
  212. package/modules/use-cases/test-module-auth.js +72 -0
  213. package/modules/utils/map-module-dto.js +18 -0
  214. package/package.json +82 -50
  215. package/prisma-mongodb/schema.prisma +362 -0
  216. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  217. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  218. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  219. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  220. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  221. package/prisma-postgresql/schema.prisma +345 -0
  222. package/queues/queuer-util.js +27 -22
  223. package/syncs/manager.js +468 -443
  224. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  225. package/syncs/repositories/sync-repository-factory.js +43 -0
  226. package/syncs/repositories/sync-repository-interface.js +109 -0
  227. package/syncs/repositories/sync-repository-mongo.js +239 -0
  228. package/syncs/repositories/sync-repository-postgres.js +319 -0
  229. package/syncs/sync.js +0 -1
  230. package/token/repositories/token-repository-documentdb.js +137 -0
  231. package/token/repositories/token-repository-factory.js +40 -0
  232. package/token/repositories/token-repository-interface.js +131 -0
  233. package/token/repositories/token-repository-mongo.js +219 -0
  234. package/token/repositories/token-repository-postgres.js +264 -0
  235. package/token/repositories/token-repository.js +219 -0
  236. package/types/core/index.d.ts +2 -2
  237. package/types/integrations/index.d.ts +2 -6
  238. package/types/module-plugin/index.d.ts +5 -59
  239. package/types/syncs/index.d.ts +0 -2
  240. package/user/repositories/user-repository-documentdb.js +441 -0
  241. package/user/repositories/user-repository-factory.js +52 -0
  242. package/user/repositories/user-repository-interface.js +201 -0
  243. package/user/repositories/user-repository-mongo.js +308 -0
  244. package/user/repositories/user-repository-postgres.js +360 -0
  245. package/user/tests/doubles/test-user-repository.js +72 -0
  246. package/user/use-cases/authenticate-user.js +127 -0
  247. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  248. package/user/use-cases/create-individual-user.js +61 -0
  249. package/user/use-cases/create-organization-user.js +47 -0
  250. package/user/use-cases/create-token-for-user-id.js +30 -0
  251. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  252. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  253. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  254. package/user/use-cases/login-user.js +122 -0
  255. package/user/user.js +125 -0
  256. package/utils/backend-path.js +38 -0
  257. package/utils/index.js +6 -0
  258. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  259. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  260. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  261. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  262. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  263. package/websocket/repositories/websocket-connection-repository.js +161 -0
  264. package/database/models/State.js +0 -9
  265. package/database/models/Token.js +0 -70
  266. package/database/mongo.js +0 -45
  267. package/encrypt/Cryptor.test.js +0 -32
  268. package/encrypt/encrypt.js +0 -132
  269. package/encrypt/encrypt.test.js +0 -1069
  270. package/errors/base-error.test.js +0 -32
  271. package/errors/fetch-error.test.js +0 -79
  272. package/errors/halt-error.test.js +0 -11
  273. package/errors/validation-errors.test.js +0 -120
  274. package/integrations/create-frigg-backend.js +0 -31
  275. package/integrations/integration-factory.js +0 -251
  276. package/integrations/integration-mapping.js +0 -43
  277. package/integrations/integration-model.js +0 -46
  278. package/integrations/integration-user.js +0 -144
  279. package/integrations/test/integration-base.test.js +0 -144
  280. package/lambda/TimeoutCatcher.test.js +0 -68
  281. package/logs/logger.test.js +0 -76
  282. package/module-plugin/auther.js +0 -393
  283. package/module-plugin/credential.js +0 -22
  284. package/module-plugin/entity-manager.js +0 -70
  285. package/module-plugin/manager.js +0 -169
  286. package/module-plugin/module-factory.js +0 -61
  287. package/module-plugin/requester/api-key.js +0 -36
  288. package/module-plugin/requester/oauth-2.js +0 -219
  289. package/module-plugin/requester/requester.test.js +0 -28
  290. package/module-plugin/test/auther.test.js +0 -97
  291. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  292. /package/{module-plugin → modules}/requester/basic.js +0 -0
  293. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
package/index.js CHANGED
@@ -24,8 +24,47 @@ const {
24
24
  Token,
25
25
  UserModel,
26
26
  WebsocketConnection,
27
+ prisma,
28
+ TokenRepository,
29
+ WebsocketConnectionRepository,
27
30
  } = require('./database/index');
28
- const { Encrypt, Cryptor } = require('./encrypt/encrypt');
31
+ const {
32
+ createUserRepository,
33
+ UserRepositoryMongo,
34
+ UserRepositoryPostgres,
35
+ } = require('./user/repositories/user-repository-factory');
36
+ const {
37
+ GetUserFromXFriggHeaders,
38
+ } = require('./user/use-cases/get-user-from-x-frigg-headers');
39
+ const {
40
+ GetUserFromAdopterJwt,
41
+ } = require('./user/use-cases/get-user-from-adopter-jwt');
42
+ const {
43
+ AuthenticateUser,
44
+ } = require('./user/use-cases/authenticate-user');
45
+
46
+ const {
47
+ CredentialRepository,
48
+ } = require('./credential/repositories/credential-repository');
49
+ const {
50
+ ModuleRepository,
51
+ } = require('./modules/repositories/module-repository');
52
+ const {
53
+ IntegrationMappingRepository,
54
+ } = require('./integrations/repositories/integration-mapping-repository');
55
+ const {
56
+ CreateProcess,
57
+ } = require('./integrations/use-cases/create-process');
58
+ const {
59
+ UpdateProcessState,
60
+ } = require('./integrations/use-cases/update-process-state');
61
+ const {
62
+ UpdateProcessMetrics,
63
+ } = require('./integrations/use-cases/update-process-metrics');
64
+ const {
65
+ GetProcess,
66
+ } = require('./integrations/use-cases/get-process');
67
+ const { Cryptor } = require('./encrypt');
29
68
  const {
30
69
  BaseError,
31
70
  FetchError,
@@ -35,30 +74,26 @@ const {
35
74
  } = require('./errors/index');
36
75
  const {
37
76
  IntegrationBase,
38
- IntegrationModel,
39
77
  Options,
40
- IntegrationMapping,
41
- IntegrationFactory,
42
- IntegrationHelper,
43
78
  createIntegrationRouter,
44
79
  checkRequiredParams,
45
- createFriggBackend,
80
+ getModulesDefinitionFromIntegrationClasses,
81
+ LoadIntegrationContextUseCase,
46
82
  } = require('./integrations/index');
47
83
  const { TimeoutCatcher } = require('./lambda/index');
48
84
  const { debug, initDebugLog, flushDebugLog } = require('./logs/index');
49
85
  const {
50
86
  Credential,
51
- EntityManager,
52
87
  Entity,
53
- ModuleManager,
54
88
  ApiKeyRequester,
55
89
  BasicAuthRequester,
56
90
  OAuth2Requester,
57
91
  Requester,
58
92
  ModuleConstants,
59
93
  ModuleFactory,
60
- Auther,
61
- } = require('./module-plugin/index');
94
+ } = require('./modules/index');
95
+ const application = require('./application');
96
+ const utils = require('./utils');
62
97
 
63
98
  // const {Sync } = require('./syncs/model');
64
99
 
@@ -91,9 +126,18 @@ module.exports = {
91
126
  Token,
92
127
  UserModel,
93
128
  WebsocketConnection,
94
-
95
- // encrypt
96
- Encrypt,
129
+ prisma,
130
+ TokenRepository,
131
+ WebsocketConnectionRepository,
132
+ createUserRepository,
133
+ UserRepositoryMongo,
134
+ UserRepositoryPostgres,
135
+ GetUserFromXFriggHeaders,
136
+ GetUserFromAdopterJwt,
137
+ AuthenticateUser,
138
+ CredentialRepository,
139
+ ModuleRepository,
140
+ IntegrationMappingRepository,
97
141
  Cryptor,
98
142
 
99
143
  // errors
@@ -105,14 +149,27 @@ module.exports = {
105
149
 
106
150
  // integrations
107
151
  IntegrationBase,
108
- IntegrationModel,
109
152
  Options,
110
- IntegrationMapping,
111
- IntegrationFactory,
112
- IntegrationHelper,
113
153
  checkRequiredParams,
114
154
  createIntegrationRouter,
115
- createFriggBackend,
155
+ getModulesDefinitionFromIntegrationClasses,
156
+ LoadIntegrationContextUseCase,
157
+ CreateProcess,
158
+ UpdateProcessState,
159
+ UpdateProcessMetrics,
160
+ GetProcess,
161
+
162
+ // application - Command factories for integration developers
163
+ application,
164
+ createFriggCommands: application.createFriggCommands,
165
+ createIntegrationCommands: application.createIntegrationCommands,
166
+ createUserCommands: application.createUserCommands,
167
+ createEntityCommands: application.createEntityCommands,
168
+ createCredentialCommands: application.createCredentialCommands,
169
+ createSchedulerCommands: application.createSchedulerCommands,
170
+ findIntegrationContextByExternalEntityId:
171
+ application.findIntegrationContextByExternalEntityId,
172
+ integrationCommands: application.integrationCommands,
116
173
 
117
174
  // lambda
118
175
  TimeoutCatcher,
@@ -124,17 +181,16 @@ module.exports = {
124
181
 
125
182
  // module plugin
126
183
  Credential,
127
- EntityManager,
128
184
  Entity,
129
- ModuleManager,
130
185
  ApiKeyRequester,
131
186
  BasicAuthRequester,
132
187
  OAuth2Requester,
133
188
  Requester,
134
189
  ModuleConstants,
135
190
  ModuleFactory,
136
- Auther,
137
-
138
191
  // queues
139
192
  QueuerUtil,
193
+
194
+ // utils
195
+ ...utils,
140
196
  };
@@ -0,0 +1,184 @@
1
+ /**
2
+ * EventBridge Scheduler Adapter
3
+ *
4
+ * Infrastructure Layer - Hexagonal Architecture
5
+ *
6
+ * Responsible for:
7
+ * - Creating one-time EventBridge Scheduler schedules
8
+ * - Deleting schedules when no longer needed
9
+ * - Checking schedule status
10
+ *
11
+ * This adapter implements SchedulerServiceInterface for AWS EventBridge Scheduler.
12
+ */
13
+
14
+ const {
15
+ SchedulerClient,
16
+ CreateScheduleCommand,
17
+ DeleteScheduleCommand,
18
+ GetScheduleCommand,
19
+ ResourceNotFoundException,
20
+ } = require('@aws-sdk/client-scheduler');
21
+
22
+ const { SchedulerServiceInterface } = require('./scheduler-service-interface');
23
+
24
+ class EventBridgeSchedulerAdapter extends SchedulerServiceInterface {
25
+ constructor({ region } = {}) {
26
+ super();
27
+ this.client = new SchedulerClient({
28
+ region: region || process.env.AWS_REGION || 'us-east-1',
29
+ });
30
+ this.scheduleGroupName =
31
+ process.env.SCHEDULE_GROUP_NAME || 'frigg-integration-schedules';
32
+ this.roleArn = process.env.SCHEDULER_ROLE_ARN;
33
+ }
34
+
35
+ /**
36
+ * Create a one-time schedule that sends a message to SQS
37
+ *
38
+ * @param {Object} params
39
+ * @param {string} params.scheduleName - Unique name for the schedule
40
+ * @param {Date} params.scheduleAt - When to trigger the schedule
41
+ * @param {string} params.queueResourceId - Queue resource identifier (ARN) to send message to
42
+ * @param {Object} params.payload - Message payload
43
+ * @returns {Promise<{scheduledJobId: string, scheduledAt: string}>}
44
+ */
45
+ async scheduleOneTime({ scheduleName, scheduleAt, queueResourceId, payload }) {
46
+ if (!scheduleName) {
47
+ throw new Error('scheduleName is required');
48
+ }
49
+ if (!scheduleAt || !(scheduleAt instanceof Date)) {
50
+ throw new Error('scheduleAt must be a valid Date object');
51
+ }
52
+ if (!queueResourceId) {
53
+ throw new Error('queueResourceId is required');
54
+ }
55
+ if (!this.roleArn) {
56
+ throw new Error(
57
+ 'SCHEDULER_ROLE_ARN environment variable is not set'
58
+ );
59
+ }
60
+
61
+ // Format date to AWS schedule expression (at(yyyy-mm-ddThh:mm:ss))
62
+ const scheduleExpression = `at(${scheduleAt.toISOString().replace(/\.\d{3}Z$/, '')})`;
63
+
64
+ const command = new CreateScheduleCommand({
65
+ Name: scheduleName,
66
+ GroupName: this.scheduleGroupName,
67
+ ScheduleExpression: scheduleExpression,
68
+ ScheduleExpressionTimezone: 'UTC',
69
+ FlexibleTimeWindow: {
70
+ Mode: 'OFF',
71
+ },
72
+ Target: {
73
+ Arn: queueResourceId,
74
+ RoleArn: this.roleArn,
75
+ Input: JSON.stringify(payload),
76
+ },
77
+ ActionAfterCompletion: 'DELETE', // Auto-cleanup after execution
78
+ });
79
+
80
+ try {
81
+ const response = await this.client.send(command);
82
+ console.log(
83
+ `[Scheduler] Created schedule ${scheduleName} for ${scheduleAt.toISOString()}`
84
+ );
85
+
86
+ return {
87
+ scheduledJobId: response.ScheduleArn,
88
+ scheduledAt: scheduleAt.toISOString(),
89
+ };
90
+ } catch (error) {
91
+ console.error(
92
+ `[Scheduler] Failed to create schedule ${scheduleName}:`,
93
+ error.message
94
+ );
95
+ throw error;
96
+ }
97
+ }
98
+
99
+ /**
100
+ * Delete a schedule
101
+ *
102
+ * @param {string} scheduleName - Name of the schedule to delete
103
+ * @returns {Promise<void>}
104
+ */
105
+ async deleteSchedule(scheduleName) {
106
+ if (!scheduleName) {
107
+ throw new Error('scheduleName is required');
108
+ }
109
+
110
+ const command = new DeleteScheduleCommand({
111
+ Name: scheduleName,
112
+ GroupName: this.scheduleGroupName,
113
+ });
114
+
115
+ try {
116
+ await this.client.send(command);
117
+ console.log(`[Scheduler] Deleted schedule ${scheduleName}`);
118
+ } catch (error) {
119
+ if (error instanceof ResourceNotFoundException) {
120
+ console.log(
121
+ `[Scheduler] Schedule ${scheduleName} not found (already deleted or executed)`
122
+ );
123
+ return; // Graceful handling - schedule doesn't exist
124
+ }
125
+ console.error(
126
+ `[Scheduler] Failed to delete schedule ${scheduleName}:`,
127
+ error.message
128
+ );
129
+ throw error;
130
+ }
131
+ }
132
+
133
+ /**
134
+ * Get schedule status
135
+ *
136
+ * @param {string} scheduleName - Name of the schedule
137
+ * @returns {Promise<{exists: boolean, scheduledAt?: string, state?: string}>}
138
+ */
139
+ async getScheduleStatus(scheduleName) {
140
+ if (!scheduleName) {
141
+ throw new Error('scheduleName is required');
142
+ }
143
+
144
+ const command = new GetScheduleCommand({
145
+ Name: scheduleName,
146
+ GroupName: this.scheduleGroupName,
147
+ });
148
+
149
+ try {
150
+ const response = await this.client.send(command);
151
+
152
+ // Parse the schedule expression to get the scheduled time
153
+ // Format: at(yyyy-mm-ddThh:mm:ss)
154
+ let scheduledAt = null;
155
+ if (response.ScheduleExpression) {
156
+ const match = response.ScheduleExpression.match(
157
+ /^at\((.+)\)$/
158
+ );
159
+ if (match) {
160
+ scheduledAt = new Date(match[1] + 'Z').toISOString();
161
+ }
162
+ }
163
+
164
+ return {
165
+ exists: true,
166
+ scheduledAt,
167
+ state: response.State,
168
+ };
169
+ } catch (error) {
170
+ if (error instanceof ResourceNotFoundException) {
171
+ return {
172
+ exists: false,
173
+ };
174
+ }
175
+ console.error(
176
+ `[Scheduler] Failed to get schedule ${scheduleName}:`,
177
+ error.message
178
+ );
179
+ throw error;
180
+ }
181
+ }
182
+ }
183
+
184
+ module.exports = { EventBridgeSchedulerAdapter };
@@ -0,0 +1,33 @@
1
+ /**
2
+ * Scheduler Infrastructure
3
+ *
4
+ * Provides scheduling capabilities for one-time jobs.
5
+ * Follows hexagonal architecture with interface + adapters pattern.
6
+ *
7
+ * Providers:
8
+ * - eventbridge: AWS EventBridge Scheduler (production)
9
+ * - mock: In-memory mock scheduler (local development)
10
+ */
11
+
12
+ const { SchedulerServiceInterface } = require('./scheduler-service-interface');
13
+ const { EventBridgeSchedulerAdapter } = require('./eventbridge-scheduler-adapter');
14
+ const { MockSchedulerAdapter } = require('./mock-scheduler-adapter');
15
+ const {
16
+ createSchedulerService,
17
+ SCHEDULER_PROVIDERS,
18
+ determineProvider,
19
+ } = require('./scheduler-service-factory');
20
+
21
+ module.exports = {
22
+ // Interface (Port)
23
+ SchedulerServiceInterface,
24
+
25
+ // Adapters
26
+ EventBridgeSchedulerAdapter,
27
+ MockSchedulerAdapter,
28
+
29
+ // Factory
30
+ createSchedulerService,
31
+ SCHEDULER_PROVIDERS,
32
+ determineProvider,
33
+ };
@@ -0,0 +1,143 @@
1
+ /**
2
+ * Mock Scheduler Adapter for Local Development
3
+ *
4
+ * Stores schedules in memory and logs instead of creating real EventBridge schedules.
5
+ * Used when SCHEDULER_PROVIDER=mock or in local/dev/test environments.
6
+ *
7
+ * This adapter implements SchedulerServiceInterface for local development and testing.
8
+ */
9
+
10
+ const { SchedulerServiceInterface } = require('./scheduler-service-interface');
11
+
12
+ class MockSchedulerAdapter extends SchedulerServiceInterface {
13
+ constructor(options = {}) {
14
+ super();
15
+ this.verbose = options.verbose || false;
16
+ this.schedules = new Map();
17
+ }
18
+
19
+ /**
20
+ * Schedule a one-time job to be executed at a specific time
21
+ *
22
+ * @param {Object} params
23
+ * @param {string} params.scheduleName - Unique name for the schedule
24
+ * @param {Date} params.scheduleAt - When to trigger the schedule
25
+ * @param {string} params.queueResourceId - Queue resource identifier to send message to
26
+ * @param {Object} params.payload - JSON payload to send
27
+ * @returns {Promise<{scheduledJobId: string, scheduledAt: string}>}
28
+ */
29
+ async scheduleOneTime({ scheduleName, scheduleAt, queueResourceId, payload }) {
30
+ if (!scheduleName) {
31
+ throw new Error('scheduleName is required');
32
+ }
33
+ if (!scheduleAt || !(scheduleAt instanceof Date)) {
34
+ throw new Error('scheduleAt must be a valid Date object');
35
+ }
36
+ if (!queueResourceId) {
37
+ throw new Error('queueResourceId is required');
38
+ }
39
+
40
+ const scheduleData = {
41
+ scheduleName,
42
+ scheduledAt: scheduleAt.toISOString(),
43
+ queueResourceId,
44
+ payload,
45
+ createdAt: new Date().toISOString(),
46
+ state: 'ENABLED',
47
+ };
48
+
49
+ this.schedules.set(scheduleName, scheduleData);
50
+
51
+ console.log(`[MockScheduler] Created schedule: ${scheduleName}`);
52
+ console.log(`[MockScheduler] Scheduled for: ${scheduleAt.toISOString()}`);
53
+ console.log(`[MockScheduler] Target: ${queueResourceId}`);
54
+ if (this.verbose) {
55
+ console.log(`[MockScheduler] Payload:`, JSON.stringify(payload, null, 2));
56
+ }
57
+
58
+ return {
59
+ scheduledJobId: `mock-job-${scheduleName}`,
60
+ scheduledAt: scheduleAt.toISOString(),
61
+ };
62
+ }
63
+
64
+ /**
65
+ * Delete a scheduled job
66
+ *
67
+ * @param {string} scheduleName - Name of the schedule to delete
68
+ * @returns {Promise<void>}
69
+ */
70
+ async deleteSchedule(scheduleName) {
71
+ if (!scheduleName) {
72
+ throw new Error('scheduleName is required');
73
+ }
74
+
75
+ const existed = this.schedules.has(scheduleName);
76
+ this.schedules.delete(scheduleName);
77
+
78
+ console.log(`[MockScheduler] Deleted schedule: ${scheduleName} (existed: ${existed})`);
79
+ }
80
+
81
+ /**
82
+ * Get the status of a scheduled job
83
+ *
84
+ * @param {string} scheduleName - Name of the schedule
85
+ * @returns {Promise<{exists: boolean, scheduledAt?: string, state?: string}>}
86
+ */
87
+ async getScheduleStatus(scheduleName) {
88
+ if (!scheduleName) {
89
+ throw new Error('scheduleName is required');
90
+ }
91
+
92
+ const schedule = this.schedules.get(scheduleName);
93
+
94
+ if (!schedule) {
95
+ return { exists: false };
96
+ }
97
+
98
+ return {
99
+ exists: true,
100
+ scheduledAt: schedule.scheduledAt,
101
+ state: schedule.state,
102
+ };
103
+ }
104
+
105
+ /**
106
+ * Get all scheduled jobs (helper for testing)
107
+ *
108
+ * @returns {Object} Map of all schedules as plain object
109
+ */
110
+ _getSchedules() {
111
+ return Object.fromEntries(this.schedules);
112
+ }
113
+
114
+ /**
115
+ * Clear all schedules (helper for testing)
116
+ */
117
+ _clearSchedules() {
118
+ const count = this.schedules.size;
119
+ this.schedules.clear();
120
+ console.log(`[MockScheduler] Cleared ${count} schedules`);
121
+ }
122
+
123
+ /**
124
+ * Simulate triggering a schedule (helper for testing)
125
+ *
126
+ * @param {string} scheduleName - Name of the schedule to trigger
127
+ * @returns {Object|null} The payload that would be sent, or null if not found
128
+ */
129
+ _simulateTrigger(scheduleName) {
130
+ const schedule = this.schedules.get(scheduleName);
131
+ if (!schedule) {
132
+ console.log(`[MockScheduler] Cannot trigger - schedule not found: ${scheduleName}`);
133
+ return null;
134
+ }
135
+
136
+ console.log(`[MockScheduler] Simulating trigger for: ${scheduleName}`);
137
+ console.log(`[MockScheduler] Payload:`, JSON.stringify(schedule.payload, null, 2));
138
+
139
+ return schedule.payload;
140
+ }
141
+ }
142
+
143
+ module.exports = { MockSchedulerAdapter };
@@ -0,0 +1,73 @@
1
+ /**
2
+ * Scheduler Service Factory
3
+ *
4
+ * Creates scheduler service instances based on configuration.
5
+ * Returns implementations of SchedulerServiceInterface.
6
+ *
7
+ * Environment Detection:
8
+ * - SCHEDULER_PROVIDER=eventbridge -> Use AWS EventBridge Scheduler
9
+ * - SCHEDULER_PROVIDER=mock -> Use in-memory mock scheduler
10
+ * - Default in dev/test/local stages -> Mock scheduler
11
+ * - Default in other stages -> EventBridge scheduler
12
+ */
13
+
14
+ const { EventBridgeSchedulerAdapter } = require('./eventbridge-scheduler-adapter');
15
+ const { MockSchedulerAdapter } = require('./mock-scheduler-adapter');
16
+
17
+ const SCHEDULER_PROVIDERS = {
18
+ EVENTBRIDGE: 'eventbridge',
19
+ MOCK: 'mock',
20
+ };
21
+
22
+ const LOCAL_STAGES = ['dev', 'test', 'local'];
23
+
24
+ /**
25
+ * Determine the scheduler provider based on environment
26
+ *
27
+ * @returns {string} Provider name
28
+ */
29
+ function determineProvider() {
30
+ const explicitProvider = process.env.SCHEDULER_PROVIDER;
31
+ if (explicitProvider) {
32
+ return explicitProvider;
33
+ }
34
+
35
+ const stage = process.env.STAGE || 'dev';
36
+ if (LOCAL_STAGES.includes(stage)) {
37
+ return SCHEDULER_PROVIDERS.MOCK;
38
+ }
39
+
40
+ return SCHEDULER_PROVIDERS.EVENTBRIDGE;
41
+ }
42
+
43
+ /**
44
+ * Create a scheduler service instance
45
+ *
46
+ * @param {Object} options
47
+ * @param {string} options.provider - Scheduler provider ('eventbridge' or 'mock')
48
+ * @param {string} options.region - AWS region (for EventBridge)
49
+ * @param {boolean} options.verbose - Verbose logging (for Mock)
50
+ * @returns {SchedulerServiceInterface} Implementation of scheduler interface
51
+ */
52
+ function createSchedulerService(options = {}) {
53
+ const provider = options.provider || determineProvider();
54
+
55
+ switch (provider) {
56
+ case SCHEDULER_PROVIDERS.EVENTBRIDGE:
57
+ return new EventBridgeSchedulerAdapter({
58
+ region: options.region,
59
+ });
60
+ case SCHEDULER_PROVIDERS.MOCK:
61
+ return new MockSchedulerAdapter({
62
+ verbose: options.verbose,
63
+ });
64
+ default:
65
+ throw new Error(`Unknown scheduler provider: ${provider}`);
66
+ }
67
+ }
68
+
69
+ module.exports = {
70
+ createSchedulerService,
71
+ SCHEDULER_PROVIDERS,
72
+ determineProvider,
73
+ };
@@ -0,0 +1,47 @@
1
+ /**
2
+ * Scheduler Service Interface (Port)
3
+ *
4
+ * Defines the contract for scheduling one-time jobs.
5
+ * All scheduler adapters must extend this interface.
6
+ *
7
+ * Following Frigg's hexagonal architecture pattern:
8
+ * - Port defines WHAT the service does (contract)
9
+ * - Adapters implement HOW (AWS EventBridge, Mock, etc.)
10
+ */
11
+ class SchedulerServiceInterface {
12
+ /**
13
+ * Schedule a one-time job to be executed at a specific time
14
+ *
15
+ * @param {Object} params
16
+ * @param {string} params.scheduleName - Unique name for the schedule
17
+ * @param {Date} params.scheduleAt - When to trigger the schedule
18
+ * @param {string} params.queueResourceId - Queue resource identifier to send message to
19
+ * @param {Object} params.payload - JSON payload to send
20
+ * @returns {Promise<{scheduledJobId: string, scheduledAt: string}>}
21
+ */
22
+ async scheduleOneTime({ scheduleName, scheduleAt, queueResourceId, payload }) {
23
+ throw new Error('Method scheduleOneTime must be implemented by subclass');
24
+ }
25
+
26
+ /**
27
+ * Delete a scheduled job
28
+ *
29
+ * @param {string} scheduleName - Name of the schedule to delete
30
+ * @returns {Promise<void>}
31
+ */
32
+ async deleteSchedule(scheduleName) {
33
+ throw new Error('Method deleteSchedule must be implemented by subclass');
34
+ }
35
+
36
+ /**
37
+ * Get the status of a scheduled job
38
+ *
39
+ * @param {string} scheduleName - Name of the schedule
40
+ * @returns {Promise<{exists: boolean, scheduledAt?: string, state?: string}>}
41
+ */
42
+ async getScheduleStatus(scheduleName) {
43
+ throw new Error('Method getScheduleStatus must be implemented by subclass');
44
+ }
45
+ }
46
+
47
+ module.exports = { SchedulerServiceInterface };