@paperclipai/server 0.2.2

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 (317) hide show
  1. package/dist/adapters/codex-models.d.ts +4 -0
  2. package/dist/adapters/codex-models.d.ts.map +1 -0
  3. package/dist/adapters/codex-models.js +98 -0
  4. package/dist/adapters/codex-models.js.map +1 -0
  5. package/dist/adapters/http/execute.d.ts +3 -0
  6. package/dist/adapters/http/execute.d.ts.map +1 -0
  7. package/dist/adapters/http/execute.js +39 -0
  8. package/dist/adapters/http/execute.js.map +1 -0
  9. package/dist/adapters/http/index.d.ts +3 -0
  10. package/dist/adapters/http/index.d.ts.map +1 -0
  11. package/dist/adapters/http/index.js +20 -0
  12. package/dist/adapters/http/index.js.map +1 -0
  13. package/dist/adapters/http/test.d.ts +3 -0
  14. package/dist/adapters/http/test.d.ts.map +1 -0
  15. package/dist/adapters/http/test.js +106 -0
  16. package/dist/adapters/http/test.js.map +1 -0
  17. package/dist/adapters/index.d.ts +4 -0
  18. package/dist/adapters/index.d.ts.map +1 -0
  19. package/dist/adapters/index.js +3 -0
  20. package/dist/adapters/index.js.map +1 -0
  21. package/dist/adapters/process/execute.d.ts +3 -0
  22. package/dist/adapters/process/execute.d.ts.map +1 -0
  23. package/dist/adapters/process/execute.js +63 -0
  24. package/dist/adapters/process/execute.js.map +1 -0
  25. package/dist/adapters/process/index.d.ts +3 -0
  26. package/dist/adapters/process/index.d.ts.map +1 -0
  27. package/dist/adapters/process/index.js +23 -0
  28. package/dist/adapters/process/index.js.map +1 -0
  29. package/dist/adapters/process/test.d.ts +3 -0
  30. package/dist/adapters/process/test.d.ts.map +1 -0
  31. package/dist/adapters/process/test.js +77 -0
  32. package/dist/adapters/process/test.js.map +1 -0
  33. package/dist/adapters/registry.d.ts +9 -0
  34. package/dist/adapters/registry.d.ts.map +1 -0
  35. package/dist/adapters/registry.js +63 -0
  36. package/dist/adapters/registry.js.map +1 -0
  37. package/dist/adapters/types.d.ts +2 -0
  38. package/dist/adapters/types.d.ts.map +1 -0
  39. package/dist/adapters/types.js +2 -0
  40. package/dist/adapters/types.js.map +1 -0
  41. package/dist/adapters/utils.d.ts +10 -0
  42. package/dist/adapters/utils.d.ts.map +1 -0
  43. package/dist/adapters/utils.js +14 -0
  44. package/dist/adapters/utils.js.map +1 -0
  45. package/dist/agent-auth-jwt.d.ts +14 -0
  46. package/dist/agent-auth-jwt.d.ts.map +1 -0
  47. package/dist/agent-auth-jwt.js +117 -0
  48. package/dist/agent-auth-jwt.js.map +1 -0
  49. package/dist/app.d.ts +20 -0
  50. package/dist/app.d.ts.map +1 -0
  51. package/dist/app.js +127 -0
  52. package/dist/app.js.map +1 -0
  53. package/dist/auth/better-auth.d.ts +23 -0
  54. package/dist/auth/better-auth.d.ts.map +1 -0
  55. package/dist/auth/better-auth.js +80 -0
  56. package/dist/auth/better-auth.js.map +1 -0
  57. package/dist/board-claim.d.ts +23 -0
  58. package/dist/board-claim.d.ts.map +1 -0
  59. package/dist/board-claim.js +115 -0
  60. package/dist/board-claim.js.map +1 -0
  61. package/dist/config-file.d.ts +3 -0
  62. package/dist/config-file.d.ts.map +1 -0
  63. package/dist/config-file.js +16 -0
  64. package/dist/config-file.js.map +1 -0
  65. package/dist/config.d.ts +33 -0
  66. package/dist/config.d.ts.map +1 -0
  67. package/dist/config.js +114 -0
  68. package/dist/config.js.map +1 -0
  69. package/dist/errors.d.ts +12 -0
  70. package/dist/errors.d.ts.map +1 -0
  71. package/dist/errors.js +28 -0
  72. package/dist/errors.js.map +1 -0
  73. package/dist/home-paths.d.ts +11 -0
  74. package/dist/home-paths.d.ts.map +1 -0
  75. package/dist/home-paths.js +54 -0
  76. package/dist/home-paths.js.map +1 -0
  77. package/dist/index.d.ts +2 -0
  78. package/dist/index.d.ts.map +1 -0
  79. package/dist/index.js +439 -0
  80. package/dist/index.js.map +1 -0
  81. package/dist/middleware/auth.d.ts +12 -0
  82. package/dist/middleware/auth.d.ts.map +1 -0
  83. package/dist/middleware/auth.js +124 -0
  84. package/dist/middleware/auth.js.map +1 -0
  85. package/dist/middleware/board-mutation-guard.d.ts +3 -0
  86. package/dist/middleware/board-mutation-guard.d.ts.map +1 -0
  87. package/dist/middleware/board-mutation-guard.js +60 -0
  88. package/dist/middleware/board-mutation-guard.js.map +1 -0
  89. package/dist/middleware/error-handler.d.ts +3 -0
  90. package/dist/middleware/error-handler.d.ts.map +1 -0
  91. package/dist/middleware/error-handler.js +22 -0
  92. package/dist/middleware/error-handler.js.map +1 -0
  93. package/dist/middleware/index.d.ts +4 -0
  94. package/dist/middleware/index.d.ts.map +1 -0
  95. package/dist/middleware/index.js +4 -0
  96. package/dist/middleware/index.js.map +1 -0
  97. package/dist/middleware/logger.d.ts +4 -0
  98. package/dist/middleware/logger.d.ts.map +1 -0
  99. package/dist/middleware/logger.js +37 -0
  100. package/dist/middleware/logger.js.map +1 -0
  101. package/dist/middleware/private-hostname-guard.d.ts +11 -0
  102. package/dist/middleware/private-hostname-guard.d.ts.map +1 -0
  103. package/dist/middleware/private-hostname-guard.js +78 -0
  104. package/dist/middleware/private-hostname-guard.js.map +1 -0
  105. package/dist/middleware/validate.d.ts +4 -0
  106. package/dist/middleware/validate.d.ts.map +1 -0
  107. package/dist/middleware/validate.js +7 -0
  108. package/dist/middleware/validate.js.map +1 -0
  109. package/dist/paths.d.ts +3 -0
  110. package/dist/paths.d.ts.map +1 -0
  111. package/dist/paths.js +31 -0
  112. package/dist/paths.js.map +1 -0
  113. package/dist/realtime/live-events-ws.d.ts +10 -0
  114. package/dist/realtime/live-events-ws.d.ts.map +1 -0
  115. package/dist/realtime/live-events-ws.js +185 -0
  116. package/dist/realtime/live-events-ws.js.map +1 -0
  117. package/dist/redaction.d.ts +4 -0
  118. package/dist/redaction.d.ts.map +1 -0
  119. package/dist/redaction.js +63 -0
  120. package/dist/redaction.js.map +1 -0
  121. package/dist/routes/access.d.ts +9 -0
  122. package/dist/routes/access.d.ts.map +1 -0
  123. package/dist/routes/access.js +887 -0
  124. package/dist/routes/access.js.map +1 -0
  125. package/dist/routes/activity.d.ts +3 -0
  126. package/dist/routes/activity.d.ts.map +1 -0
  127. package/dist/routes/activity.js +87 -0
  128. package/dist/routes/activity.js.map +1 -0
  129. package/dist/routes/agents.d.ts +3 -0
  130. package/dist/routes/agents.d.ts.map +1 -0
  131. package/dist/routes/agents.js +1132 -0
  132. package/dist/routes/agents.js.map +1 -0
  133. package/dist/routes/approvals.d.ts +3 -0
  134. package/dist/routes/approvals.d.ts.map +1 -0
  135. package/dist/routes/approvals.js +271 -0
  136. package/dist/routes/approvals.js.map +1 -0
  137. package/dist/routes/assets.d.ts +4 -0
  138. package/dist/routes/assets.d.ts.map +1 -0
  139. package/dist/routes/assets.js +138 -0
  140. package/dist/routes/assets.js.map +1 -0
  141. package/dist/routes/authz.d.ts +15 -0
  142. package/dist/routes/authz.d.ts.map +1 -0
  143. package/dist/routes/authz.js +40 -0
  144. package/dist/routes/authz.js.map +1 -0
  145. package/dist/routes/companies.d.ts +3 -0
  146. package/dist/routes/companies.d.ts.map +1 -0
  147. package/dist/routes/companies.js +159 -0
  148. package/dist/routes/companies.js.map +1 -0
  149. package/dist/routes/costs.d.ts +3 -0
  150. package/dist/routes/costs.d.ts.map +1 -0
  151. package/dist/routes/costs.js +113 -0
  152. package/dist/routes/costs.js.map +1 -0
  153. package/dist/routes/dashboard.d.ts +3 -0
  154. package/dist/routes/dashboard.d.ts.map +1 -0
  155. package/dist/routes/dashboard.js +15 -0
  156. package/dist/routes/dashboard.js.map +1 -0
  157. package/dist/routes/goals.d.ts +3 -0
  158. package/dist/routes/goals.d.ts.map +1 -0
  159. package/dist/routes/goals.js +95 -0
  160. package/dist/routes/goals.js.map +1 -0
  161. package/dist/routes/health.d.ts +9 -0
  162. package/dist/routes/health.d.ts.map +1 -0
  163. package/dist/routes/health.js +38 -0
  164. package/dist/routes/health.js.map +1 -0
  165. package/dist/routes/index.d.ts +15 -0
  166. package/dist/routes/index.d.ts.map +1 -0
  167. package/dist/routes/index.js +15 -0
  168. package/dist/routes/index.js.map +1 -0
  169. package/dist/routes/issues.d.ts +4 -0
  170. package/dist/routes/issues.d.ts.map +1 -0
  171. package/dist/routes/issues.js +973 -0
  172. package/dist/routes/issues.js.map +1 -0
  173. package/dist/routes/llms.d.ts +3 -0
  174. package/dist/routes/llms.d.ts.map +1 -0
  175. package/dist/routes/llms.js +78 -0
  176. package/dist/routes/llms.js.map +1 -0
  177. package/dist/routes/projects.d.ts +3 -0
  178. package/dist/routes/projects.d.ts.map +1 -0
  179. package/dist/routes/projects.js +253 -0
  180. package/dist/routes/projects.js.map +1 -0
  181. package/dist/routes/secrets.d.ts +3 -0
  182. package/dist/routes/secrets.d.ts.map +1 -0
  183. package/dist/routes/secrets.js +128 -0
  184. package/dist/routes/secrets.js.map +1 -0
  185. package/dist/routes/sidebar-badges.d.ts +3 -0
  186. package/dist/routes/sidebar-badges.d.ts.map +1 -0
  187. package/dist/routes/sidebar-badges.js +47 -0
  188. package/dist/routes/sidebar-badges.js.map +1 -0
  189. package/dist/secrets/external-stub-providers.d.ts +5 -0
  190. package/dist/secrets/external-stub-providers.d.ts.map +1 -0
  191. package/dist/secrets/external-stub-providers.js +21 -0
  192. package/dist/secrets/external-stub-providers.js.map +1 -0
  193. package/dist/secrets/local-encrypted-provider.d.ts +3 -0
  194. package/dist/secrets/local-encrypted-provider.d.ts.map +1 -0
  195. package/dist/secrets/local-encrypted-provider.js +116 -0
  196. package/dist/secrets/local-encrypted-provider.js.map +1 -0
  197. package/dist/secrets/provider-registry.d.ts +5 -0
  198. package/dist/secrets/provider-registry.d.ts.map +1 -0
  199. package/dist/secrets/provider-registry.js +20 -0
  200. package/dist/secrets/provider-registry.js.map +1 -0
  201. package/dist/secrets/types.d.ts +21 -0
  202. package/dist/secrets/types.d.ts.map +1 -0
  203. package/dist/secrets/types.js +2 -0
  204. package/dist/secrets/types.js.map +1 -0
  205. package/dist/services/access.d.ts +81 -0
  206. package/dist/services/access.d.ts.map +1 -0
  207. package/dist/services/access.js +187 -0
  208. package/dist/services/access.js.map +1 -0
  209. package/dist/services/activity-log.d.ts +14 -0
  210. package/dist/services/activity-log.d.ts.map +1 -0
  211. package/dist/services/activity-log.js +32 -0
  212. package/dist/services/activity-log.js.map +1 -0
  213. package/dist/services/activity.d.ts +764 -0
  214. package/dist/services/activity.d.ts.map +1 -0
  215. package/dist/services/activity.js +105 -0
  216. package/dist/services/activity.js.map +1 -0
  217. package/dist/services/agent-permissions.d.ts +6 -0
  218. package/dist/services/agent-permissions.d.ts.map +1 -0
  219. package/dist/services/agent-permissions.js +18 -0
  220. package/dist/services/agent-permissions.js.map +1 -0
  221. package/dist/services/agents.d.ts +1494 -0
  222. package/dist/services/agents.d.ts.map +1 -0
  223. package/dist/services/agents.js +454 -0
  224. package/dist/services/agents.js.map +1 -0
  225. package/dist/services/approvals.d.ts +540 -0
  226. package/dist/services/approvals.d.ts.map +1 -0
  227. package/dist/services/approvals.js +173 -0
  228. package/dist/services/approvals.js.map +1 -0
  229. package/dist/services/assets.d.ts +33 -0
  230. package/dist/services/assets.d.ts.map +1 -0
  231. package/dist/services/assets.js +17 -0
  232. package/dist/services/assets.js.map +1 -0
  233. package/dist/services/companies.d.ts +503 -0
  234. package/dist/services/companies.d.ts.map +1 -0
  235. package/dist/services/companies.js +120 -0
  236. package/dist/services/companies.js.map +1 -0
  237. package/dist/services/company-portability.d.ts +8 -0
  238. package/dist/services/company-portability.d.ts.map +1 -0
  239. package/dist/services/company-portability.js +851 -0
  240. package/dist/services/company-portability.js.map +1 -0
  241. package/dist/services/costs.d.ts +50 -0
  242. package/dist/services/costs.d.ts.map +1 -0
  243. package/dist/services/costs.js +166 -0
  244. package/dist/services/costs.js.map +1 -0
  245. package/dist/services/dashboard.d.ts +21 -0
  246. package/dist/services/dashboard.d.ts.map +1 -0
  247. package/dist/services/dashboard.js +96 -0
  248. package/dist/services/dashboard.js.map +1 -0
  249. package/dist/services/goals.d.ts +407 -0
  250. package/dist/services/goals.d.ts.map +1 -0
  251. package/dist/services/goals.js +29 -0
  252. package/dist/services/goals.js.map +1 -0
  253. package/dist/services/heartbeat.d.ts +1666 -0
  254. package/dist/services/heartbeat.d.ts.map +1 -0
  255. package/dist/services/heartbeat.js +1752 -0
  256. package/dist/services/heartbeat.js.map +1 -0
  257. package/dist/services/index.d.ts +20 -0
  258. package/dist/services/index.d.ts.map +1 -0
  259. package/dist/services/index.js +20 -0
  260. package/dist/services/index.js.map +1 -0
  261. package/dist/services/issue-approvals.d.ts +56 -0
  262. package/dist/services/issue-approvals.d.ts.map +1 -0
  263. package/dist/services/issue-approvals.js +153 -0
  264. package/dist/services/issue-approvals.js.map +1 -0
  265. package/dist/services/issues.d.ts +756 -0
  266. package/dist/services/issues.d.ts.map +1 -0
  267. package/dist/services/issues.js +917 -0
  268. package/dist/services/issues.js.map +1 -0
  269. package/dist/services/live-events.d.ts +12 -0
  270. package/dist/services/live-events.d.ts.map +1 -0
  271. package/dist/services/live-events.js +24 -0
  272. package/dist/services/live-events.js.map +1 -0
  273. package/dist/services/projects.d.ts +66 -0
  274. package/dist/services/projects.d.ts.map +1 -0
  275. package/dist/services/projects.js +472 -0
  276. package/dist/services/projects.js.map +1 -0
  277. package/dist/services/run-log-store.d.ts +34 -0
  278. package/dist/services/run-log-store.d.ts.map +1 -0
  279. package/dist/services/run-log-store.js +112 -0
  280. package/dist/services/run-log-store.js.map +1 -0
  281. package/dist/services/secrets.d.ts +506 -0
  282. package/dist/services/secrets.d.ts.map +1 -0
  283. package/dist/services/secrets.js +284 -0
  284. package/dist/services/secrets.js.map +1 -0
  285. package/dist/services/sidebar-badges.d.ts +9 -0
  286. package/dist/services/sidebar-badges.d.ts.map +1 -0
  287. package/dist/services/sidebar-badges.js +33 -0
  288. package/dist/services/sidebar-badges.js.map +1 -0
  289. package/dist/startup-banner.d.ts +27 -0
  290. package/dist/startup-banner.d.ts.map +1 -0
  291. package/dist/startup-banner.js +112 -0
  292. package/dist/startup-banner.js.map +1 -0
  293. package/dist/storage/index.d.ts +6 -0
  294. package/dist/storage/index.d.ts.map +1 -0
  295. package/dist/storage/index.js +29 -0
  296. package/dist/storage/index.js.map +1 -0
  297. package/dist/storage/local-disk-provider.d.ts +3 -0
  298. package/dist/storage/local-disk-provider.d.ts.map +1 -0
  299. package/dist/storage/local-disk-provider.js +79 -0
  300. package/dist/storage/local-disk-provider.js.map +1 -0
  301. package/dist/storage/provider-registry.d.ts +4 -0
  302. package/dist/storage/provider-registry.d.ts.map +1 -0
  303. package/dist/storage/provider-registry.js +15 -0
  304. package/dist/storage/provider-registry.js.map +1 -0
  305. package/dist/storage/s3-provider.d.ts +11 -0
  306. package/dist/storage/s3-provider.d.ts.map +1 -0
  307. package/dist/storage/s3-provider.js +123 -0
  308. package/dist/storage/s3-provider.js.map +1 -0
  309. package/dist/storage/service.d.ts +3 -0
  310. package/dist/storage/service.d.ts.map +1 -0
  311. package/dist/storage/service.js +120 -0
  312. package/dist/storage/service.js.map +1 -0
  313. package/dist/storage/types.d.ts +55 -0
  314. package/dist/storage/types.d.ts.map +1 -0
  315. package/dist/storage/types.js +2 -0
  316. package/dist/storage/types.js.map +1 -0
  317. package/package.json +62 -0
@@ -0,0 +1,284 @@
1
+ import { and, desc, eq } from "drizzle-orm";
2
+ import { companySecrets, companySecretVersions } from "@paperclipai/db";
3
+ import { envBindingSchema } from "@paperclipai/shared";
4
+ import { conflict, notFound, unprocessable } from "../errors.js";
5
+ import { getSecretProvider, listSecretProviders } from "../secrets/provider-registry.js";
6
+ const ENV_KEY_RE = /^[A-Za-z_][A-Za-z0-9_]*$/;
7
+ const SENSITIVE_ENV_KEY_RE = /(api[-_]?key|access[-_]?token|auth(?:_?token)?|authorization|bearer|secret|passwd|password|credential|jwt|private[-_]?key|cookie|connectionstring)/i;
8
+ const REDACTED_SENTINEL = "***REDACTED***";
9
+ function asRecord(value) {
10
+ if (typeof value !== "object" || value === null || Array.isArray(value))
11
+ return null;
12
+ return value;
13
+ }
14
+ function isSensitiveEnvKey(key) {
15
+ return SENSITIVE_ENV_KEY_RE.test(key);
16
+ }
17
+ function canonicalizeBinding(binding) {
18
+ if (typeof binding === "string") {
19
+ return { type: "plain", value: binding };
20
+ }
21
+ if (binding.type === "plain") {
22
+ return { type: "plain", value: String(binding.value) };
23
+ }
24
+ return {
25
+ type: "secret_ref",
26
+ secretId: binding.secretId,
27
+ version: binding.version ?? "latest",
28
+ };
29
+ }
30
+ export function secretService(db) {
31
+ async function getById(id) {
32
+ return db
33
+ .select()
34
+ .from(companySecrets)
35
+ .where(eq(companySecrets.id, id))
36
+ .then((rows) => rows[0] ?? null);
37
+ }
38
+ async function getByName(companyId, name) {
39
+ return db
40
+ .select()
41
+ .from(companySecrets)
42
+ .where(and(eq(companySecrets.companyId, companyId), eq(companySecrets.name, name)))
43
+ .then((rows) => rows[0] ?? null);
44
+ }
45
+ async function getSecretVersion(secretId, version) {
46
+ return db
47
+ .select()
48
+ .from(companySecretVersions)
49
+ .where(and(eq(companySecretVersions.secretId, secretId), eq(companySecretVersions.version, version)))
50
+ .then((rows) => rows[0] ?? null);
51
+ }
52
+ async function assertSecretInCompany(companyId, secretId) {
53
+ const secret = await getById(secretId);
54
+ if (!secret)
55
+ throw notFound("Secret not found");
56
+ if (secret.companyId !== companyId)
57
+ throw unprocessable("Secret must belong to same company");
58
+ return secret;
59
+ }
60
+ async function resolveSecretValue(companyId, secretId, version) {
61
+ const secret = await assertSecretInCompany(companyId, secretId);
62
+ const resolvedVersion = version === "latest" ? secret.latestVersion : version;
63
+ const versionRow = await getSecretVersion(secret.id, resolvedVersion);
64
+ if (!versionRow)
65
+ throw notFound("Secret version not found");
66
+ const provider = getSecretProvider(secret.provider);
67
+ return provider.resolveVersion({
68
+ material: versionRow.material,
69
+ externalRef: secret.externalRef,
70
+ });
71
+ }
72
+ async function normalizeEnvConfig(companyId, envValue, opts) {
73
+ const record = asRecord(envValue);
74
+ if (!record)
75
+ throw unprocessable("adapterConfig.env must be an object");
76
+ const normalized = {};
77
+ for (const [key, rawBinding] of Object.entries(record)) {
78
+ if (!ENV_KEY_RE.test(key)) {
79
+ throw unprocessable(`Invalid environment variable name: ${key}`);
80
+ }
81
+ const parsed = envBindingSchema.safeParse(rawBinding);
82
+ if (!parsed.success) {
83
+ throw unprocessable(`Invalid environment binding for key: ${key}`);
84
+ }
85
+ const binding = canonicalizeBinding(parsed.data);
86
+ if (binding.type === "plain") {
87
+ if (opts?.strictMode && isSensitiveEnvKey(key) && binding.value.trim().length > 0) {
88
+ throw unprocessable(`Strict secret mode requires secret references for sensitive key: ${key}`);
89
+ }
90
+ if (binding.value === REDACTED_SENTINEL) {
91
+ throw unprocessable(`Refusing to persist redacted placeholder for key: ${key}`);
92
+ }
93
+ normalized[key] = binding;
94
+ continue;
95
+ }
96
+ await assertSecretInCompany(companyId, binding.secretId);
97
+ normalized[key] = {
98
+ type: "secret_ref",
99
+ secretId: binding.secretId,
100
+ version: binding.version,
101
+ };
102
+ }
103
+ return normalized;
104
+ }
105
+ async function normalizeAdapterConfigForPersistenceInternal(companyId, adapterConfig, opts) {
106
+ const normalized = { ...adapterConfig };
107
+ if (!Object.prototype.hasOwnProperty.call(adapterConfig, "env")) {
108
+ return normalized;
109
+ }
110
+ normalized.env = await normalizeEnvConfig(companyId, adapterConfig.env, opts);
111
+ return normalized;
112
+ }
113
+ return {
114
+ listProviders: () => listSecretProviders(),
115
+ list: (companyId) => db
116
+ .select()
117
+ .from(companySecrets)
118
+ .where(eq(companySecrets.companyId, companyId))
119
+ .orderBy(desc(companySecrets.createdAt)),
120
+ getById,
121
+ getByName,
122
+ create: async (companyId, input, actor) => {
123
+ const existing = await getByName(companyId, input.name);
124
+ if (existing)
125
+ throw conflict(`Secret already exists: ${input.name}`);
126
+ const provider = getSecretProvider(input.provider);
127
+ const prepared = await provider.createVersion({
128
+ value: input.value,
129
+ externalRef: input.externalRef ?? null,
130
+ });
131
+ return db.transaction(async (tx) => {
132
+ const secret = await tx
133
+ .insert(companySecrets)
134
+ .values({
135
+ companyId,
136
+ name: input.name,
137
+ provider: input.provider,
138
+ externalRef: prepared.externalRef,
139
+ latestVersion: 1,
140
+ description: input.description ?? null,
141
+ createdByAgentId: actor?.agentId ?? null,
142
+ createdByUserId: actor?.userId ?? null,
143
+ })
144
+ .returning()
145
+ .then((rows) => rows[0]);
146
+ await tx.insert(companySecretVersions).values({
147
+ secretId: secret.id,
148
+ version: 1,
149
+ material: prepared.material,
150
+ valueSha256: prepared.valueSha256,
151
+ createdByAgentId: actor?.agentId ?? null,
152
+ createdByUserId: actor?.userId ?? null,
153
+ });
154
+ return secret;
155
+ });
156
+ },
157
+ rotate: async (secretId, input, actor) => {
158
+ const secret = await getById(secretId);
159
+ if (!secret)
160
+ throw notFound("Secret not found");
161
+ const provider = getSecretProvider(secret.provider);
162
+ const nextVersion = secret.latestVersion + 1;
163
+ const prepared = await provider.createVersion({
164
+ value: input.value,
165
+ externalRef: input.externalRef ?? secret.externalRef ?? null,
166
+ });
167
+ return db.transaction(async (tx) => {
168
+ await tx.insert(companySecretVersions).values({
169
+ secretId: secret.id,
170
+ version: nextVersion,
171
+ material: prepared.material,
172
+ valueSha256: prepared.valueSha256,
173
+ createdByAgentId: actor?.agentId ?? null,
174
+ createdByUserId: actor?.userId ?? null,
175
+ });
176
+ const updated = await tx
177
+ .update(companySecrets)
178
+ .set({
179
+ latestVersion: nextVersion,
180
+ externalRef: prepared.externalRef,
181
+ updatedAt: new Date(),
182
+ })
183
+ .where(eq(companySecrets.id, secret.id))
184
+ .returning()
185
+ .then((rows) => rows[0] ?? null);
186
+ if (!updated)
187
+ throw notFound("Secret not found");
188
+ return updated;
189
+ });
190
+ },
191
+ update: async (secretId, patch) => {
192
+ const secret = await getById(secretId);
193
+ if (!secret)
194
+ throw notFound("Secret not found");
195
+ if (patch.name && patch.name !== secret.name) {
196
+ const duplicate = await getByName(secret.companyId, patch.name);
197
+ if (duplicate && duplicate.id !== secret.id) {
198
+ throw conflict(`Secret already exists: ${patch.name}`);
199
+ }
200
+ }
201
+ return db
202
+ .update(companySecrets)
203
+ .set({
204
+ name: patch.name ?? secret.name,
205
+ description: patch.description === undefined ? secret.description : patch.description,
206
+ externalRef: patch.externalRef === undefined ? secret.externalRef : patch.externalRef,
207
+ updatedAt: new Date(),
208
+ })
209
+ .where(eq(companySecrets.id, secret.id))
210
+ .returning()
211
+ .then((rows) => rows[0] ?? null);
212
+ },
213
+ remove: async (secretId) => {
214
+ const secret = await getById(secretId);
215
+ if (!secret)
216
+ return null;
217
+ await db.delete(companySecrets).where(eq(companySecrets.id, secretId));
218
+ return secret;
219
+ },
220
+ normalizeAdapterConfigForPersistence: async (companyId, adapterConfig, opts) => normalizeAdapterConfigForPersistenceInternal(companyId, adapterConfig, opts),
221
+ normalizeHireApprovalPayloadForPersistence: async (companyId, payload, opts) => {
222
+ const normalized = { ...payload };
223
+ const adapterConfig = asRecord(payload.adapterConfig);
224
+ if (adapterConfig) {
225
+ normalized.adapterConfig = await normalizeAdapterConfigForPersistenceInternal(companyId, adapterConfig, opts);
226
+ }
227
+ return normalized;
228
+ },
229
+ resolveEnvBindings: async (companyId, envValue) => {
230
+ const record = asRecord(envValue);
231
+ if (!record)
232
+ return {};
233
+ const resolved = {};
234
+ for (const [key, rawBinding] of Object.entries(record)) {
235
+ if (!ENV_KEY_RE.test(key)) {
236
+ throw unprocessable(`Invalid environment variable name: ${key}`);
237
+ }
238
+ const parsed = envBindingSchema.safeParse(rawBinding);
239
+ if (!parsed.success) {
240
+ throw unprocessable(`Invalid environment binding for key: ${key}`);
241
+ }
242
+ const binding = canonicalizeBinding(parsed.data);
243
+ if (binding.type === "plain") {
244
+ resolved[key] = binding.value;
245
+ }
246
+ else {
247
+ resolved[key] = await resolveSecretValue(companyId, binding.secretId, binding.version);
248
+ }
249
+ }
250
+ return resolved;
251
+ },
252
+ resolveAdapterConfigForRuntime: async (companyId, adapterConfig) => {
253
+ const resolved = { ...adapterConfig };
254
+ if (!Object.prototype.hasOwnProperty.call(adapterConfig, "env")) {
255
+ return resolved;
256
+ }
257
+ const record = asRecord(adapterConfig.env);
258
+ if (!record) {
259
+ resolved.env = {};
260
+ return resolved;
261
+ }
262
+ const env = {};
263
+ for (const [key, rawBinding] of Object.entries(record)) {
264
+ if (!ENV_KEY_RE.test(key)) {
265
+ throw unprocessable(`Invalid environment variable name: ${key}`);
266
+ }
267
+ const parsed = envBindingSchema.safeParse(rawBinding);
268
+ if (!parsed.success) {
269
+ throw unprocessable(`Invalid environment binding for key: ${key}`);
270
+ }
271
+ const binding = canonicalizeBinding(parsed.data);
272
+ if (binding.type === "plain") {
273
+ env[key] = binding.value;
274
+ }
275
+ else {
276
+ env[key] = await resolveSecretValue(companyId, binding.secretId, binding.version);
277
+ }
278
+ }
279
+ resolved.env = env;
280
+ return resolved;
281
+ },
282
+ };
283
+ }
284
+ //# sourceMappingURL=secrets.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"secrets.js","sourceRoot":"","sources":["../../src/services/secrets.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { Db } from "@paperclipai/db";
2
+ import type { SidebarBadges } from "@paperclipai/shared";
3
+ export declare function sidebarBadgeService(db: Db): {
4
+ get: (companyId: string, extra?: {
5
+ joinRequests?: number;
6
+ assignedIssues?: number;
7
+ }) => Promise<SidebarBadges>;
8
+ };
9
+ //# sourceMappingURL=sidebar-badges.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sidebar-badges.d.ts","sourceRoot":"","sources":["../../src/services/sidebar-badges.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,EAAE,EAAE,MAAM,iBAAiB,CAAC;AAE1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAKzD,wBAAgB,mBAAmB,CAAC,EAAE,EAAE,EAAE;qBAGzB,MAAM,UACT;QAAE,YAAY,CAAC,EAAE,MAAM,CAAC;QAAC,cAAc,CAAC,EAAE,MAAM,CAAA;KAAE,KACzD,OAAO,CAAC,aAAa,CAAC;EAyC5B"}
@@ -0,0 +1,33 @@
1
+ import { and, desc, eq, inArray, not, sql } from "drizzle-orm";
2
+ import { agents, approvals, heartbeatRuns } from "@paperclipai/db";
3
+ const ACTIONABLE_APPROVAL_STATUSES = ["pending", "revision_requested"];
4
+ const FAILED_HEARTBEAT_STATUSES = ["failed", "timed_out"];
5
+ export function sidebarBadgeService(db) {
6
+ return {
7
+ get: async (companyId, extra) => {
8
+ const actionableApprovals = await db
9
+ .select({ count: sql `count(*)` })
10
+ .from(approvals)
11
+ .where(and(eq(approvals.companyId, companyId), inArray(approvals.status, ACTIONABLE_APPROVAL_STATUSES)))
12
+ .then((rows) => Number(rows[0]?.count ?? 0));
13
+ const latestRunByAgent = await db
14
+ .selectDistinctOn([heartbeatRuns.agentId], {
15
+ runStatus: heartbeatRuns.status,
16
+ })
17
+ .from(heartbeatRuns)
18
+ .innerJoin(agents, eq(heartbeatRuns.agentId, agents.id))
19
+ .where(and(eq(heartbeatRuns.companyId, companyId), eq(agents.companyId, companyId), not(eq(agents.status, "terminated"))))
20
+ .orderBy(heartbeatRuns.agentId, desc(heartbeatRuns.createdAt));
21
+ const failedRuns = latestRunByAgent.filter((row) => FAILED_HEARTBEAT_STATUSES.includes(row.runStatus)).length;
22
+ const joinRequests = extra?.joinRequests ?? 0;
23
+ const assignedIssues = extra?.assignedIssues ?? 0;
24
+ return {
25
+ inbox: actionableApprovals + failedRuns + joinRequests + assignedIssues,
26
+ approvals: actionableApprovals,
27
+ failedRuns,
28
+ joinRequests,
29
+ };
30
+ },
31
+ };
32
+ }
33
+ //# sourceMappingURL=sidebar-badges.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sidebar-badges.js","sourceRoot":"","sources":["../../src/services/sidebar-badges.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAE/D,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAGnE,MAAM,4BAA4B,GAAG,CAAC,SAAS,EAAE,oBAAoB,CAAC,CAAC;AACvE,MAAM,yBAAyB,GAAG,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;AAE1D,MAAM,UAAU,mBAAmB,CAAC,EAAM;IACxC,OAAO;QACL,GAAG,EAAE,KAAK,EACR,SAAiB,EACjB,KAA0D,EAClC,EAAE;YAC1B,MAAM,mBAAmB,GAAG,MAAM,EAAE;iBACjC,MAAM,CAAC,EAAE,KAAK,EAAE,GAAG,CAAQ,UAAU,EAAE,CAAC;iBACxC,IAAI,CAAC,SAAS,CAAC;iBACf,KAAK,CACJ,GAAG,CACD,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,SAAS,CAAC,EAClC,OAAO,CAAC,SAAS,CAAC,MAAM,EAAE,4BAA4B,CAAC,CACxD,CACF;iBACA,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC;YAE/C,MAAM,gBAAgB,GAAG,MAAM,EAAE;iBAC9B,gBAAgB,CAAC,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE;gBACzC,SAAS,EAAE,aAAa,CAAC,MAAM;aAChC,CAAC;iBACD,IAAI,CAAC,aAAa,CAAC;iBACnB,SAAS,CAAC,MAAM,EAAE,EAAE,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;iBACvD,KAAK,CACJ,GAAG,CACD,EAAE,CAAC,aAAa,CAAC,SAAS,EAAE,SAAS,CAAC,EACtC,EAAE,CAAC,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,EAC/B,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC,CACrC,CACF;iBACA,OAAO,CAAC,aAAa,CAAC,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC;YAEjE,MAAM,UAAU,GAAG,gBAAgB,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CACjD,yBAAyB,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAClD,CAAC,MAAM,CAAC;YAET,MAAM,YAAY,GAAG,KAAK,EAAE,YAAY,IAAI,CAAC,CAAC;YAC9C,MAAM,cAAc,GAAG,KAAK,EAAE,cAAc,IAAI,CAAC,CAAC;YAClD,OAAO;gBACL,KAAK,EAAE,mBAAmB,GAAG,UAAU,GAAG,YAAY,GAAG,cAAc;gBACvE,SAAS,EAAE,mBAAmB;gBAC9B,UAAU;gBACV,YAAY;aACb,CAAC;QACJ,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,27 @@
1
+ import type { DeploymentExposure, DeploymentMode } from "@paperclipai/shared";
2
+ type UiMode = "none" | "static" | "vite-dev";
3
+ type ExternalPostgresInfo = {
4
+ mode: "external-postgres";
5
+ connectionString: string;
6
+ };
7
+ type EmbeddedPostgresInfo = {
8
+ mode: "embedded-postgres";
9
+ dataDir: string;
10
+ port: number;
11
+ };
12
+ type StartupBannerOptions = {
13
+ host: string;
14
+ deploymentMode: DeploymentMode;
15
+ deploymentExposure: DeploymentExposure;
16
+ authReady: boolean;
17
+ requestedPort: number;
18
+ listenPort: number;
19
+ uiMode: UiMode;
20
+ db: ExternalPostgresInfo | EmbeddedPostgresInfo;
21
+ migrationSummary: string;
22
+ heartbeatSchedulerEnabled: boolean;
23
+ heartbeatSchedulerIntervalMs: number;
24
+ };
25
+ export declare function printStartupBanner(opts: StartupBannerOptions): void;
26
+ export {};
27
+ //# sourceMappingURL=startup-banner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"startup-banner.d.ts","sourceRoot":"","sources":["../src/startup-banner.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,kBAAkB,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAI9E,KAAK,MAAM,GAAG,MAAM,GAAG,QAAQ,GAAG,UAAU,CAAC;AAE7C,KAAK,oBAAoB,GAAG;IAC1B,IAAI,EAAE,mBAAmB,CAAC;IAC1B,gBAAgB,EAAE,MAAM,CAAC;CAC1B,CAAC;AAEF,KAAK,oBAAoB,GAAG;IAC1B,IAAI,EAAE,mBAAmB,CAAC;IAC1B,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;CACd,CAAC;AAEF,KAAK,oBAAoB,GAAG;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,cAAc,EAAE,cAAc,CAAC;IAC/B,kBAAkB,EAAE,kBAAkB,CAAC;IACvC,SAAS,EAAE,OAAO,CAAC;IACnB,aAAa,EAAE,MAAM,CAAC;IACtB,UAAU,EAAE,MAAM,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;IACf,EAAE,EAAE,oBAAoB,GAAG,oBAAoB,CAAC;IAChD,gBAAgB,EAAE,MAAM,CAAC;IACzB,yBAAyB,EAAE,OAAO,CAAC;IACnC,4BAA4B,EAAE,MAAM,CAAC;CACtC,CAAC;AA+DF,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,oBAAoB,GAAG,IAAI,CAuEnE"}
@@ -0,0 +1,112 @@
1
+ import { existsSync, readFileSync } from "node:fs";
2
+ import { resolvePaperclipConfigPath, resolvePaperclipEnvPath } from "./paths.js";
3
+ import { parse as parseEnvFileContents } from "dotenv";
4
+ const ansi = {
5
+ reset: "\x1b[0m",
6
+ bold: "\x1b[1m",
7
+ dim: "\x1b[2m",
8
+ cyan: "\x1b[36m",
9
+ green: "\x1b[32m",
10
+ yellow: "\x1b[33m",
11
+ magenta: "\x1b[35m",
12
+ blue: "\x1b[34m",
13
+ };
14
+ function color(text, c) {
15
+ return `${ansi[c]}${text}${ansi.reset}`;
16
+ }
17
+ function row(label, value) {
18
+ return `${color(label.padEnd(16), "dim")} ${value}`;
19
+ }
20
+ function redactConnectionString(raw) {
21
+ try {
22
+ const u = new URL(raw);
23
+ const user = u.username || "user";
24
+ const auth = `${user}:***@`;
25
+ return `${u.protocol}//${auth}${u.host}${u.pathname}`;
26
+ }
27
+ catch {
28
+ return "<invalid DATABASE_URL>";
29
+ }
30
+ }
31
+ function resolveAgentJwtSecretStatus(envFilePath) {
32
+ const envValue = process.env.PAPERCLIP_AGENT_JWT_SECRET?.trim();
33
+ if (envValue) {
34
+ return {
35
+ status: "pass",
36
+ message: "set",
37
+ };
38
+ }
39
+ if (existsSync(envFilePath)) {
40
+ const parsed = parseEnvFileContents(readFileSync(envFilePath, "utf-8"));
41
+ const fileValue = typeof parsed.PAPERCLIP_AGENT_JWT_SECRET === "string" ? parsed.PAPERCLIP_AGENT_JWT_SECRET.trim() : "";
42
+ if (fileValue) {
43
+ return {
44
+ status: "warn",
45
+ message: `found in ${envFilePath} but not loaded`,
46
+ };
47
+ }
48
+ }
49
+ return {
50
+ status: "warn",
51
+ message: "missing (run `pnpm paperclipai onboard`)",
52
+ };
53
+ }
54
+ export function printStartupBanner(opts) {
55
+ const baseHost = opts.host === "0.0.0.0" ? "localhost" : opts.host;
56
+ const baseUrl = `http://${baseHost}:${opts.listenPort}`;
57
+ const apiUrl = `${baseUrl}/api`;
58
+ const uiUrl = opts.uiMode === "none" ? "disabled" : baseUrl;
59
+ const configPath = resolvePaperclipConfigPath();
60
+ const envFilePath = resolvePaperclipEnvPath();
61
+ const agentJwtSecret = resolveAgentJwtSecretStatus(envFilePath);
62
+ const dbMode = opts.db.mode === "embedded-postgres"
63
+ ? color("embedded-postgres", "green")
64
+ : color("external-postgres", "yellow");
65
+ const uiMode = opts.uiMode === "vite-dev"
66
+ ? color("vite-dev-middleware", "cyan")
67
+ : opts.uiMode === "static"
68
+ ? color("static-ui", "magenta")
69
+ : color("headless-api", "yellow");
70
+ const portValue = opts.requestedPort === opts.listenPort
71
+ ? `${opts.listenPort}`
72
+ : `${opts.listenPort} ${color(`(requested ${opts.requestedPort})`, "dim")}`;
73
+ const dbDetails = opts.db.mode === "embedded-postgres"
74
+ ? `${opts.db.dataDir} ${color(`(pg:${opts.db.port})`, "dim")}`
75
+ : redactConnectionString(opts.db.connectionString);
76
+ const heartbeat = opts.heartbeatSchedulerEnabled
77
+ ? `enabled ${color(`(${opts.heartbeatSchedulerIntervalMs}ms)`, "dim")}`
78
+ : color("disabled", "yellow");
79
+ const art = [
80
+ color("██████╗ █████╗ ██████╗ ███████╗██████╗ ██████╗██╗ ██╗██████╗ ", "cyan"),
81
+ color("██╔══██╗██╔══██╗██╔══██╗██╔════╝██╔══██╗██╔════╝██║ ██║██╔══██╗", "cyan"),
82
+ color("██████╔╝███████║██████╔╝█████╗ ██████╔╝██║ ██║ ██║██████╔╝", "cyan"),
83
+ color("██╔═══╝ ██╔══██║██╔═══╝ ██╔══╝ ██╔══██╗██║ ██║ ██║██╔═══╝ ", "cyan"),
84
+ color("██║ ██║ ██║██║ ███████╗██║ ██║╚██████╗███████╗██║██║ ", "cyan"),
85
+ color("╚═╝ ╚═╝ ╚═╝╚═╝ ╚══════╝╚═╝ ╚═╝ ╚═════╝╚══════╝╚═╝╚═╝ ", "cyan"),
86
+ ];
87
+ const lines = [
88
+ "",
89
+ ...art,
90
+ color(" ───────────────────────────────────────────────────────", "blue"),
91
+ row("Mode", `${dbMode} | ${uiMode}`),
92
+ row("Deploy", `${opts.deploymentMode} (${opts.deploymentExposure})`),
93
+ row("Auth", opts.authReady ? color("ready", "green") : color("not-ready", "yellow")),
94
+ row("Server", portValue),
95
+ row("API", `${apiUrl} ${color(`(health: ${apiUrl}/health)`, "dim")}`),
96
+ row("UI", uiUrl),
97
+ row("Database", dbDetails),
98
+ row("Migrations", opts.migrationSummary),
99
+ row("Agent JWT", agentJwtSecret.status === "pass"
100
+ ? color(agentJwtSecret.message, "green")
101
+ : color(agentJwtSecret.message, "yellow")),
102
+ row("Heartbeat", heartbeat),
103
+ row("Config", configPath),
104
+ agentJwtSecret.status === "warn"
105
+ ? color(" ───────────────────────────────────────────────────────", "yellow")
106
+ : null,
107
+ color(" ───────────────────────────────────────────────────────", "blue"),
108
+ "",
109
+ ];
110
+ console.log(lines.filter((line) => line !== null).join("\n"));
111
+ }
112
+ //# sourceMappingURL=startup-banner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"startup-banner.js","sourceRoot":"","sources":["../src/startup-banner.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import { type Config } from "../config.js";
2
+ import type { StorageService } from "./types.js";
3
+ export declare function createStorageServiceFromConfig(config: Config): StorageService;
4
+ export declare function getStorageService(): StorageService;
5
+ export type { StorageService, PutFileResult } from "./types.js";
6
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAc,KAAK,MAAM,EAAE,MAAM,cAAc,CAAC;AAGvD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAiBjD,wBAAgB,8BAA8B,CAAC,MAAM,EAAE,MAAM,GAAG,cAAc,CAE7E;AAED,wBAAgB,iBAAiB,IAAI,cAAc,CAQlD;AAED,YAAY,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC"}
@@ -0,0 +1,29 @@
1
+ import { loadConfig } from "../config.js";
2
+ import { createStorageProviderFromConfig } from "./provider-registry.js";
3
+ import { createStorageService } from "./service.js";
4
+ let cachedStorageService = null;
5
+ let cachedSignature = null;
6
+ function signatureForConfig(config) {
7
+ return JSON.stringify({
8
+ provider: config.storageProvider,
9
+ localDisk: config.storageLocalDiskBaseDir,
10
+ s3Bucket: config.storageS3Bucket,
11
+ s3Region: config.storageS3Region,
12
+ s3Endpoint: config.storageS3Endpoint,
13
+ s3Prefix: config.storageS3Prefix,
14
+ s3ForcePathStyle: config.storageS3ForcePathStyle,
15
+ });
16
+ }
17
+ export function createStorageServiceFromConfig(config) {
18
+ return createStorageService(createStorageProviderFromConfig(config));
19
+ }
20
+ export function getStorageService() {
21
+ const config = loadConfig();
22
+ const signature = signatureForConfig(config);
23
+ if (!cachedStorageService || cachedSignature !== signature) {
24
+ cachedStorageService = createStorageServiceFromConfig(config);
25
+ cachedSignature = signature;
26
+ }
27
+ return cachedStorageService;
28
+ }
29
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAe,MAAM,cAAc,CAAC;AACvD,OAAO,EAAE,+BAA+B,EAAE,MAAM,wBAAwB,CAAC;AACzE,OAAO,EAAE,oBAAoB,EAAE,MAAM,cAAc,CAAC;AAGpD,IAAI,oBAAoB,GAA0B,IAAI,CAAC;AACvD,IAAI,eAAe,GAAkB,IAAI,CAAC;AAE1C,SAAS,kBAAkB,CAAC,MAAc;IACxC,OAAO,IAAI,CAAC,SAAS,CAAC;QACpB,QAAQ,EAAE,MAAM,CAAC,eAAe;QAChC,SAAS,EAAE,MAAM,CAAC,uBAAuB;QACzC,QAAQ,EAAE,MAAM,CAAC,eAAe;QAChC,QAAQ,EAAE,MAAM,CAAC,eAAe;QAChC,UAAU,EAAE,MAAM,CAAC,iBAAiB;QACpC,QAAQ,EAAE,MAAM,CAAC,eAAe;QAChC,gBAAgB,EAAE,MAAM,CAAC,uBAAuB;KACjD,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,8BAA8B,CAAC,MAAc;IAC3D,OAAO,oBAAoB,CAAC,+BAA+B,CAAC,MAAM,CAAC,CAAC,CAAC;AACvE,CAAC;AAED,MAAM,UAAU,iBAAiB;IAC/B,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,SAAS,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC;IAC7C,IAAI,CAAC,oBAAoB,IAAI,eAAe,KAAK,SAAS,EAAE,CAAC;QAC3D,oBAAoB,GAAG,8BAA8B,CAAC,MAAM,CAAC,CAAC;QAC9D,eAAe,GAAG,SAAS,CAAC;IAC9B,CAAC;IACD,OAAO,oBAAoB,CAAC;AAC9B,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { StorageProvider } from "./types.js";
2
+ export declare function createLocalDiskStorageProvider(baseDir: string): StorageProvider;
3
+ //# sourceMappingURL=local-disk-provider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local-disk-provider.d.ts","sourceRoot":"","sources":["../../src/storage/local-disk-provider.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,eAAe,EAAqC,MAAM,YAAY,CAAC;AAmCrF,wBAAgB,8BAA8B,CAAC,OAAO,EAAE,MAAM,GAAG,eAAe,CAmD/E"}
@@ -0,0 +1,79 @@
1
+ import { createReadStream, promises as fs } from "node:fs";
2
+ import path from "node:path";
3
+ import { notFound, badRequest } from "../errors.js";
4
+ function normalizeObjectKey(objectKey) {
5
+ const normalized = objectKey.replace(/\\/g, "/").trim();
6
+ if (!normalized || normalized.startsWith("/")) {
7
+ throw badRequest("Invalid object key");
8
+ }
9
+ const parts = normalized.split("/").filter((part) => part.length > 0);
10
+ if (parts.length === 0 || parts.some((part) => part === "." || part === "..")) {
11
+ throw badRequest("Invalid object key");
12
+ }
13
+ return parts.join("/");
14
+ }
15
+ function resolveWithin(baseDir, objectKey) {
16
+ const normalizedKey = normalizeObjectKey(objectKey);
17
+ const resolved = path.resolve(baseDir, normalizedKey);
18
+ const base = path.resolve(baseDir);
19
+ if (resolved !== base && !resolved.startsWith(base + path.sep)) {
20
+ throw badRequest("Invalid object key path");
21
+ }
22
+ return resolved;
23
+ }
24
+ async function statOrNull(filePath) {
25
+ try {
26
+ return await fs.stat(filePath);
27
+ }
28
+ catch {
29
+ return null;
30
+ }
31
+ }
32
+ export function createLocalDiskStorageProvider(baseDir) {
33
+ const root = path.resolve(baseDir);
34
+ return {
35
+ id: "local_disk",
36
+ async putObject(input) {
37
+ const targetPath = resolveWithin(root, input.objectKey);
38
+ const dir = path.dirname(targetPath);
39
+ await fs.mkdir(dir, { recursive: true });
40
+ const tempPath = `${targetPath}.tmp-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`;
41
+ await fs.writeFile(tempPath, input.body);
42
+ await fs.rename(tempPath, targetPath);
43
+ },
44
+ async getObject(input) {
45
+ const filePath = resolveWithin(root, input.objectKey);
46
+ const stat = await statOrNull(filePath);
47
+ if (!stat || !stat.isFile()) {
48
+ throw notFound("Object not found");
49
+ }
50
+ return {
51
+ stream: createReadStream(filePath),
52
+ contentLength: stat.size,
53
+ lastModified: stat.mtime,
54
+ };
55
+ },
56
+ async headObject(input) {
57
+ const filePath = resolveWithin(root, input.objectKey);
58
+ const stat = await statOrNull(filePath);
59
+ if (!stat || !stat.isFile()) {
60
+ return { exists: false };
61
+ }
62
+ return {
63
+ exists: true,
64
+ contentLength: stat.size,
65
+ lastModified: stat.mtime,
66
+ };
67
+ },
68
+ async deleteObject(input) {
69
+ const filePath = resolveWithin(root, input.objectKey);
70
+ try {
71
+ await fs.unlink(filePath);
72
+ }
73
+ catch {
74
+ // idempotent delete
75
+ }
76
+ },
77
+ };
78
+ }
79
+ //# sourceMappingURL=local-disk-provider.js.map