@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.
- package/dist/adapters/codex-models.d.ts +4 -0
- package/dist/adapters/codex-models.d.ts.map +1 -0
- package/dist/adapters/codex-models.js +98 -0
- package/dist/adapters/codex-models.js.map +1 -0
- package/dist/adapters/http/execute.d.ts +3 -0
- package/dist/adapters/http/execute.d.ts.map +1 -0
- package/dist/adapters/http/execute.js +39 -0
- package/dist/adapters/http/execute.js.map +1 -0
- package/dist/adapters/http/index.d.ts +3 -0
- package/dist/adapters/http/index.d.ts.map +1 -0
- package/dist/adapters/http/index.js +20 -0
- package/dist/adapters/http/index.js.map +1 -0
- package/dist/adapters/http/test.d.ts +3 -0
- package/dist/adapters/http/test.d.ts.map +1 -0
- package/dist/adapters/http/test.js +106 -0
- package/dist/adapters/http/test.js.map +1 -0
- package/dist/adapters/index.d.ts +4 -0
- package/dist/adapters/index.d.ts.map +1 -0
- package/dist/adapters/index.js +3 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/adapters/process/execute.d.ts +3 -0
- package/dist/adapters/process/execute.d.ts.map +1 -0
- package/dist/adapters/process/execute.js +63 -0
- package/dist/adapters/process/execute.js.map +1 -0
- package/dist/adapters/process/index.d.ts +3 -0
- package/dist/adapters/process/index.d.ts.map +1 -0
- package/dist/adapters/process/index.js +23 -0
- package/dist/adapters/process/index.js.map +1 -0
- package/dist/adapters/process/test.d.ts +3 -0
- package/dist/adapters/process/test.d.ts.map +1 -0
- package/dist/adapters/process/test.js +77 -0
- package/dist/adapters/process/test.js.map +1 -0
- package/dist/adapters/registry.d.ts +9 -0
- package/dist/adapters/registry.d.ts.map +1 -0
- package/dist/adapters/registry.js +63 -0
- package/dist/adapters/registry.js.map +1 -0
- package/dist/adapters/types.d.ts +2 -0
- package/dist/adapters/types.d.ts.map +1 -0
- package/dist/adapters/types.js +2 -0
- package/dist/adapters/types.js.map +1 -0
- package/dist/adapters/utils.d.ts +10 -0
- package/dist/adapters/utils.d.ts.map +1 -0
- package/dist/adapters/utils.js +14 -0
- package/dist/adapters/utils.js.map +1 -0
- package/dist/agent-auth-jwt.d.ts +14 -0
- package/dist/agent-auth-jwt.d.ts.map +1 -0
- package/dist/agent-auth-jwt.js +117 -0
- package/dist/agent-auth-jwt.js.map +1 -0
- package/dist/app.d.ts +20 -0
- package/dist/app.d.ts.map +1 -0
- package/dist/app.js +127 -0
- package/dist/app.js.map +1 -0
- package/dist/auth/better-auth.d.ts +23 -0
- package/dist/auth/better-auth.d.ts.map +1 -0
- package/dist/auth/better-auth.js +80 -0
- package/dist/auth/better-auth.js.map +1 -0
- package/dist/board-claim.d.ts +23 -0
- package/dist/board-claim.d.ts.map +1 -0
- package/dist/board-claim.js +115 -0
- package/dist/board-claim.js.map +1 -0
- package/dist/config-file.d.ts +3 -0
- package/dist/config-file.d.ts.map +1 -0
- package/dist/config-file.js +16 -0
- package/dist/config-file.js.map +1 -0
- package/dist/config.d.ts +33 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +114 -0
- package/dist/config.js.map +1 -0
- package/dist/errors.d.ts +12 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +28 -0
- package/dist/errors.js.map +1 -0
- package/dist/home-paths.d.ts +11 -0
- package/dist/home-paths.d.ts.map +1 -0
- package/dist/home-paths.js +54 -0
- package/dist/home-paths.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +439 -0
- package/dist/index.js.map +1 -0
- package/dist/middleware/auth.d.ts +12 -0
- package/dist/middleware/auth.d.ts.map +1 -0
- package/dist/middleware/auth.js +124 -0
- package/dist/middleware/auth.js.map +1 -0
- package/dist/middleware/board-mutation-guard.d.ts +3 -0
- package/dist/middleware/board-mutation-guard.d.ts.map +1 -0
- package/dist/middleware/board-mutation-guard.js +60 -0
- package/dist/middleware/board-mutation-guard.js.map +1 -0
- package/dist/middleware/error-handler.d.ts +3 -0
- package/dist/middleware/error-handler.d.ts.map +1 -0
- package/dist/middleware/error-handler.js +22 -0
- package/dist/middleware/error-handler.js.map +1 -0
- package/dist/middleware/index.d.ts +4 -0
- package/dist/middleware/index.d.ts.map +1 -0
- package/dist/middleware/index.js +4 -0
- package/dist/middleware/index.js.map +1 -0
- package/dist/middleware/logger.d.ts +4 -0
- package/dist/middleware/logger.d.ts.map +1 -0
- package/dist/middleware/logger.js +37 -0
- package/dist/middleware/logger.js.map +1 -0
- package/dist/middleware/private-hostname-guard.d.ts +11 -0
- package/dist/middleware/private-hostname-guard.d.ts.map +1 -0
- package/dist/middleware/private-hostname-guard.js +78 -0
- package/dist/middleware/private-hostname-guard.js.map +1 -0
- package/dist/middleware/validate.d.ts +4 -0
- package/dist/middleware/validate.d.ts.map +1 -0
- package/dist/middleware/validate.js +7 -0
- package/dist/middleware/validate.js.map +1 -0
- package/dist/paths.d.ts +3 -0
- package/dist/paths.d.ts.map +1 -0
- package/dist/paths.js +31 -0
- package/dist/paths.js.map +1 -0
- package/dist/realtime/live-events-ws.d.ts +10 -0
- package/dist/realtime/live-events-ws.d.ts.map +1 -0
- package/dist/realtime/live-events-ws.js +185 -0
- package/dist/realtime/live-events-ws.js.map +1 -0
- package/dist/redaction.d.ts +4 -0
- package/dist/redaction.d.ts.map +1 -0
- package/dist/redaction.js +63 -0
- package/dist/redaction.js.map +1 -0
- package/dist/routes/access.d.ts +9 -0
- package/dist/routes/access.d.ts.map +1 -0
- package/dist/routes/access.js +887 -0
- package/dist/routes/access.js.map +1 -0
- package/dist/routes/activity.d.ts +3 -0
- package/dist/routes/activity.d.ts.map +1 -0
- package/dist/routes/activity.js +87 -0
- package/dist/routes/activity.js.map +1 -0
- package/dist/routes/agents.d.ts +3 -0
- package/dist/routes/agents.d.ts.map +1 -0
- package/dist/routes/agents.js +1132 -0
- package/dist/routes/agents.js.map +1 -0
- package/dist/routes/approvals.d.ts +3 -0
- package/dist/routes/approvals.d.ts.map +1 -0
- package/dist/routes/approvals.js +271 -0
- package/dist/routes/approvals.js.map +1 -0
- package/dist/routes/assets.d.ts +4 -0
- package/dist/routes/assets.d.ts.map +1 -0
- package/dist/routes/assets.js +138 -0
- package/dist/routes/assets.js.map +1 -0
- package/dist/routes/authz.d.ts +15 -0
- package/dist/routes/authz.d.ts.map +1 -0
- package/dist/routes/authz.js +40 -0
- package/dist/routes/authz.js.map +1 -0
- package/dist/routes/companies.d.ts +3 -0
- package/dist/routes/companies.d.ts.map +1 -0
- package/dist/routes/companies.js +159 -0
- package/dist/routes/companies.js.map +1 -0
- package/dist/routes/costs.d.ts +3 -0
- package/dist/routes/costs.d.ts.map +1 -0
- package/dist/routes/costs.js +113 -0
- package/dist/routes/costs.js.map +1 -0
- package/dist/routes/dashboard.d.ts +3 -0
- package/dist/routes/dashboard.d.ts.map +1 -0
- package/dist/routes/dashboard.js +15 -0
- package/dist/routes/dashboard.js.map +1 -0
- package/dist/routes/goals.d.ts +3 -0
- package/dist/routes/goals.d.ts.map +1 -0
- package/dist/routes/goals.js +95 -0
- package/dist/routes/goals.js.map +1 -0
- package/dist/routes/health.d.ts +9 -0
- package/dist/routes/health.d.ts.map +1 -0
- package/dist/routes/health.js +38 -0
- package/dist/routes/health.js.map +1 -0
- package/dist/routes/index.d.ts +15 -0
- package/dist/routes/index.d.ts.map +1 -0
- package/dist/routes/index.js +15 -0
- package/dist/routes/index.js.map +1 -0
- package/dist/routes/issues.d.ts +4 -0
- package/dist/routes/issues.d.ts.map +1 -0
- package/dist/routes/issues.js +973 -0
- package/dist/routes/issues.js.map +1 -0
- package/dist/routes/llms.d.ts +3 -0
- package/dist/routes/llms.d.ts.map +1 -0
- package/dist/routes/llms.js +78 -0
- package/dist/routes/llms.js.map +1 -0
- package/dist/routes/projects.d.ts +3 -0
- package/dist/routes/projects.d.ts.map +1 -0
- package/dist/routes/projects.js +253 -0
- package/dist/routes/projects.js.map +1 -0
- package/dist/routes/secrets.d.ts +3 -0
- package/dist/routes/secrets.d.ts.map +1 -0
- package/dist/routes/secrets.js +128 -0
- package/dist/routes/secrets.js.map +1 -0
- package/dist/routes/sidebar-badges.d.ts +3 -0
- package/dist/routes/sidebar-badges.d.ts.map +1 -0
- package/dist/routes/sidebar-badges.js +47 -0
- package/dist/routes/sidebar-badges.js.map +1 -0
- package/dist/secrets/external-stub-providers.d.ts +5 -0
- package/dist/secrets/external-stub-providers.d.ts.map +1 -0
- package/dist/secrets/external-stub-providers.js +21 -0
- package/dist/secrets/external-stub-providers.js.map +1 -0
- package/dist/secrets/local-encrypted-provider.d.ts +3 -0
- package/dist/secrets/local-encrypted-provider.d.ts.map +1 -0
- package/dist/secrets/local-encrypted-provider.js +116 -0
- package/dist/secrets/local-encrypted-provider.js.map +1 -0
- package/dist/secrets/provider-registry.d.ts +5 -0
- package/dist/secrets/provider-registry.d.ts.map +1 -0
- package/dist/secrets/provider-registry.js +20 -0
- package/dist/secrets/provider-registry.js.map +1 -0
- package/dist/secrets/types.d.ts +21 -0
- package/dist/secrets/types.d.ts.map +1 -0
- package/dist/secrets/types.js +2 -0
- package/dist/secrets/types.js.map +1 -0
- package/dist/services/access.d.ts +81 -0
- package/dist/services/access.d.ts.map +1 -0
- package/dist/services/access.js +187 -0
- package/dist/services/access.js.map +1 -0
- package/dist/services/activity-log.d.ts +14 -0
- package/dist/services/activity-log.d.ts.map +1 -0
- package/dist/services/activity-log.js +32 -0
- package/dist/services/activity-log.js.map +1 -0
- package/dist/services/activity.d.ts +764 -0
- package/dist/services/activity.d.ts.map +1 -0
- package/dist/services/activity.js +105 -0
- package/dist/services/activity.js.map +1 -0
- package/dist/services/agent-permissions.d.ts +6 -0
- package/dist/services/agent-permissions.d.ts.map +1 -0
- package/dist/services/agent-permissions.js +18 -0
- package/dist/services/agent-permissions.js.map +1 -0
- package/dist/services/agents.d.ts +1494 -0
- package/dist/services/agents.d.ts.map +1 -0
- package/dist/services/agents.js +454 -0
- package/dist/services/agents.js.map +1 -0
- package/dist/services/approvals.d.ts +540 -0
- package/dist/services/approvals.d.ts.map +1 -0
- package/dist/services/approvals.js +173 -0
- package/dist/services/approvals.js.map +1 -0
- package/dist/services/assets.d.ts +33 -0
- package/dist/services/assets.d.ts.map +1 -0
- package/dist/services/assets.js +17 -0
- package/dist/services/assets.js.map +1 -0
- package/dist/services/companies.d.ts +503 -0
- package/dist/services/companies.d.ts.map +1 -0
- package/dist/services/companies.js +120 -0
- package/dist/services/companies.js.map +1 -0
- package/dist/services/company-portability.d.ts +8 -0
- package/dist/services/company-portability.d.ts.map +1 -0
- package/dist/services/company-portability.js +851 -0
- package/dist/services/company-portability.js.map +1 -0
- package/dist/services/costs.d.ts +50 -0
- package/dist/services/costs.d.ts.map +1 -0
- package/dist/services/costs.js +166 -0
- package/dist/services/costs.js.map +1 -0
- package/dist/services/dashboard.d.ts +21 -0
- package/dist/services/dashboard.d.ts.map +1 -0
- package/dist/services/dashboard.js +96 -0
- package/dist/services/dashboard.js.map +1 -0
- package/dist/services/goals.d.ts +407 -0
- package/dist/services/goals.d.ts.map +1 -0
- package/dist/services/goals.js +29 -0
- package/dist/services/goals.js.map +1 -0
- package/dist/services/heartbeat.d.ts +1666 -0
- package/dist/services/heartbeat.d.ts.map +1 -0
- package/dist/services/heartbeat.js +1752 -0
- package/dist/services/heartbeat.js.map +1 -0
- package/dist/services/index.d.ts +20 -0
- package/dist/services/index.d.ts.map +1 -0
- package/dist/services/index.js +20 -0
- package/dist/services/index.js.map +1 -0
- package/dist/services/issue-approvals.d.ts +56 -0
- package/dist/services/issue-approvals.d.ts.map +1 -0
- package/dist/services/issue-approvals.js +153 -0
- package/dist/services/issue-approvals.js.map +1 -0
- package/dist/services/issues.d.ts +756 -0
- package/dist/services/issues.d.ts.map +1 -0
- package/dist/services/issues.js +917 -0
- package/dist/services/issues.js.map +1 -0
- package/dist/services/live-events.d.ts +12 -0
- package/dist/services/live-events.d.ts.map +1 -0
- package/dist/services/live-events.js +24 -0
- package/dist/services/live-events.js.map +1 -0
- package/dist/services/projects.d.ts +66 -0
- package/dist/services/projects.d.ts.map +1 -0
- package/dist/services/projects.js +472 -0
- package/dist/services/projects.js.map +1 -0
- package/dist/services/run-log-store.d.ts +34 -0
- package/dist/services/run-log-store.d.ts.map +1 -0
- package/dist/services/run-log-store.js +112 -0
- package/dist/services/run-log-store.js.map +1 -0
- package/dist/services/secrets.d.ts +506 -0
- package/dist/services/secrets.d.ts.map +1 -0
- package/dist/services/secrets.js +284 -0
- package/dist/services/secrets.js.map +1 -0
- package/dist/services/sidebar-badges.d.ts +9 -0
- package/dist/services/sidebar-badges.d.ts.map +1 -0
- package/dist/services/sidebar-badges.js +33 -0
- package/dist/services/sidebar-badges.js.map +1 -0
- package/dist/startup-banner.d.ts +27 -0
- package/dist/startup-banner.d.ts.map +1 -0
- package/dist/startup-banner.js +112 -0
- package/dist/startup-banner.js.map +1 -0
- package/dist/storage/index.d.ts +6 -0
- package/dist/storage/index.d.ts.map +1 -0
- package/dist/storage/index.js +29 -0
- package/dist/storage/index.js.map +1 -0
- package/dist/storage/local-disk-provider.d.ts +3 -0
- package/dist/storage/local-disk-provider.d.ts.map +1 -0
- package/dist/storage/local-disk-provider.js +79 -0
- package/dist/storage/local-disk-provider.js.map +1 -0
- package/dist/storage/provider-registry.d.ts +4 -0
- package/dist/storage/provider-registry.d.ts.map +1 -0
- package/dist/storage/provider-registry.js +15 -0
- package/dist/storage/provider-registry.js.map +1 -0
- package/dist/storage/s3-provider.d.ts +11 -0
- package/dist/storage/s3-provider.d.ts.map +1 -0
- package/dist/storage/s3-provider.js +123 -0
- package/dist/storage/s3-provider.js.map +1 -0
- package/dist/storage/service.d.ts +3 -0
- package/dist/storage/service.d.ts.map +1 -0
- package/dist/storage/service.js +120 -0
- package/dist/storage/service.js.map +1 -0
- package/dist/storage/types.d.ts +55 -0
- package/dist/storage/types.d.ts.map +1 -0
- package/dist/storage/types.js +2 -0
- package/dist/storage/types.js.map +1 -0
- package/package.json +62 -0
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { Router } from "express";
|
|
2
|
+
import { and, eq, inArray, isNull, sql } from "drizzle-orm";
|
|
3
|
+
import { issues, joinRequests } from "@paperclipai/db";
|
|
4
|
+
import { sidebarBadgeService } from "../services/sidebar-badges.js";
|
|
5
|
+
import { accessService } from "../services/access.js";
|
|
6
|
+
import { assertCompanyAccess } from "./authz.js";
|
|
7
|
+
const INBOX_ISSUE_STATUSES = ["backlog", "todo", "in_progress", "in_review", "blocked"];
|
|
8
|
+
export function sidebarBadgeRoutes(db) {
|
|
9
|
+
const router = Router();
|
|
10
|
+
const svc = sidebarBadgeService(db);
|
|
11
|
+
const access = accessService(db);
|
|
12
|
+
router.get("/companies/:companyId/sidebar-badges", async (req, res) => {
|
|
13
|
+
const companyId = req.params.companyId;
|
|
14
|
+
assertCompanyAccess(req, companyId);
|
|
15
|
+
let canApproveJoins = false;
|
|
16
|
+
if (req.actor.type === "board") {
|
|
17
|
+
canApproveJoins =
|
|
18
|
+
req.actor.source === "local_implicit" ||
|
|
19
|
+
Boolean(req.actor.isInstanceAdmin) ||
|
|
20
|
+
(await access.canUser(companyId, req.actor.userId, "joins:approve"));
|
|
21
|
+
}
|
|
22
|
+
else if (req.actor.type === "agent" && req.actor.agentId) {
|
|
23
|
+
canApproveJoins = await access.hasPermission(companyId, "agent", req.actor.agentId, "joins:approve");
|
|
24
|
+
}
|
|
25
|
+
const joinRequestCount = canApproveJoins
|
|
26
|
+
? await db
|
|
27
|
+
.select({ count: sql `count(*)` })
|
|
28
|
+
.from(joinRequests)
|
|
29
|
+
.where(and(eq(joinRequests.companyId, companyId), eq(joinRequests.status, "pending_approval")))
|
|
30
|
+
.then((rows) => Number(rows[0]?.count ?? 0))
|
|
31
|
+
: 0;
|
|
32
|
+
const assignedIssueCount = req.actor.type === "board" && req.actor.userId
|
|
33
|
+
? await db
|
|
34
|
+
.select({ count: sql `count(*)` })
|
|
35
|
+
.from(issues)
|
|
36
|
+
.where(and(eq(issues.companyId, companyId), eq(issues.assigneeUserId, req.actor.userId), inArray(issues.status, [...INBOX_ISSUE_STATUSES]), isNull(issues.hiddenAt)))
|
|
37
|
+
.then((rows) => Number(rows[0]?.count ?? 0))
|
|
38
|
+
: 0;
|
|
39
|
+
const badges = await svc.get(companyId, {
|
|
40
|
+
joinRequests: joinRequestCount,
|
|
41
|
+
assignedIssues: assignedIssueCount,
|
|
42
|
+
});
|
|
43
|
+
res.json(badges);
|
|
44
|
+
});
|
|
45
|
+
return router;
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=sidebar-badges.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sidebar-badges.js","sourceRoot":"","sources":["../../src/routes/sidebar-badges.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { SecretProviderModule } from "./types.js";
|
|
2
|
+
export declare const awsSecretsManagerProvider: SecretProviderModule;
|
|
3
|
+
export declare const gcpSecretManagerProvider: SecretProviderModule;
|
|
4
|
+
export declare const vaultProvider: SecretProviderModule;
|
|
5
|
+
//# sourceMappingURL=external-stub-providers.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"external-stub-providers.d.ts","sourceRoot":"","sources":["../../src/secrets/external-stub-providers.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAC;AAsBvD,eAAO,MAAM,yBAAyB,sBAGrC,CAAC;AACF,eAAO,MAAM,wBAAwB,sBAGpC,CAAC;AACF,eAAO,MAAM,aAAa,sBAAkD,CAAC"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { unprocessable } from "../errors.js";
|
|
2
|
+
function unavailableProvider(id, label) {
|
|
3
|
+
return {
|
|
4
|
+
id,
|
|
5
|
+
descriptor: {
|
|
6
|
+
id,
|
|
7
|
+
label,
|
|
8
|
+
requiresExternalRef: true,
|
|
9
|
+
},
|
|
10
|
+
async createVersion() {
|
|
11
|
+
throw unprocessable(`${id} provider is not configured in this deployment`);
|
|
12
|
+
},
|
|
13
|
+
async resolveVersion() {
|
|
14
|
+
throw unprocessable(`${id} provider is not configured in this deployment`);
|
|
15
|
+
},
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
export const awsSecretsManagerProvider = unavailableProvider("aws_secrets_manager", "AWS Secrets Manager");
|
|
19
|
+
export const gcpSecretManagerProvider = unavailableProvider("gcp_secret_manager", "GCP Secret Manager");
|
|
20
|
+
export const vaultProvider = unavailableProvider("vault", "HashiCorp Vault");
|
|
21
|
+
//# sourceMappingURL=external-stub-providers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"external-stub-providers.js","sourceRoot":"","sources":["../../src/secrets/external-stub-providers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAG7C,SAAS,mBAAmB,CAC1B,EAA0D,EAC1D,KAAa;IAEb,OAAO;QACL,EAAE;QACF,UAAU,EAAE;YACV,EAAE;YACF,KAAK;YACL,mBAAmB,EAAE,IAAI;SAC1B;QACD,KAAK,CAAC,aAAa;YACjB,MAAM,aAAa,CAAC,GAAG,EAAE,gDAAgD,CAAC,CAAC;QAC7E,CAAC;QACD,KAAK,CAAC,cAAc;YAClB,MAAM,aAAa,CAAC,GAAG,EAAE,gDAAgD,CAAC,CAAC;QAC7E,CAAC;KACF,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,MAAM,yBAAyB,GAAG,mBAAmB,CAC1D,qBAAqB,EACrB,qBAAqB,CACtB,CAAC;AACF,MAAM,CAAC,MAAM,wBAAwB,GAAG,mBAAmB,CACzD,oBAAoB,EACpB,oBAAoB,CACrB,CAAC;AACF,MAAM,CAAC,MAAM,aAAa,GAAG,mBAAmB,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"local-encrypted-provider.d.ts","sourceRoot":"","sources":["../../src/secrets/local-encrypted-provider.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,oBAAoB,EAA+B,MAAM,YAAY,CAAC;AAgHpF,eAAO,MAAM,sBAAsB,EAAE,oBAmBpC,CAAC"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
import { createCipheriv, createDecipheriv, createHash, randomBytes } from "node:crypto";
|
|
2
|
+
import { mkdirSync, readFileSync, writeFileSync, existsSync, chmodSync } from "node:fs";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import { badRequest } from "../errors.js";
|
|
5
|
+
function resolveMasterKeyFilePath() {
|
|
6
|
+
const fromEnv = process.env.PAPERCLIP_SECRETS_MASTER_KEY_FILE;
|
|
7
|
+
if (fromEnv && fromEnv.trim().length > 0)
|
|
8
|
+
return path.resolve(fromEnv.trim());
|
|
9
|
+
return path.resolve(process.cwd(), "data/secrets/master.key");
|
|
10
|
+
}
|
|
11
|
+
function decodeMasterKey(raw) {
|
|
12
|
+
const trimmed = raw.trim();
|
|
13
|
+
if (!trimmed)
|
|
14
|
+
return null;
|
|
15
|
+
if (/^[A-Fa-f0-9]{64}$/.test(trimmed)) {
|
|
16
|
+
return Buffer.from(trimmed, "hex");
|
|
17
|
+
}
|
|
18
|
+
try {
|
|
19
|
+
const decoded = Buffer.from(trimmed, "base64");
|
|
20
|
+
if (decoded.length === 32)
|
|
21
|
+
return decoded;
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
// ignored
|
|
25
|
+
}
|
|
26
|
+
if (Buffer.byteLength(trimmed, "utf8") === 32) {
|
|
27
|
+
return Buffer.from(trimmed, "utf8");
|
|
28
|
+
}
|
|
29
|
+
return null;
|
|
30
|
+
}
|
|
31
|
+
function loadOrCreateMasterKey() {
|
|
32
|
+
const envKeyRaw = process.env.PAPERCLIP_SECRETS_MASTER_KEY;
|
|
33
|
+
if (envKeyRaw && envKeyRaw.trim().length > 0) {
|
|
34
|
+
const fromEnv = decodeMasterKey(envKeyRaw);
|
|
35
|
+
if (!fromEnv) {
|
|
36
|
+
throw badRequest("Invalid PAPERCLIP_SECRETS_MASTER_KEY (expected 32-byte base64, 64-char hex, or raw 32-char string)");
|
|
37
|
+
}
|
|
38
|
+
return fromEnv;
|
|
39
|
+
}
|
|
40
|
+
const keyPath = resolveMasterKeyFilePath();
|
|
41
|
+
if (existsSync(keyPath)) {
|
|
42
|
+
const raw = readFileSync(keyPath, "utf8");
|
|
43
|
+
const decoded = decodeMasterKey(raw);
|
|
44
|
+
if (!decoded) {
|
|
45
|
+
throw badRequest(`Invalid secrets master key at ${keyPath}`);
|
|
46
|
+
}
|
|
47
|
+
return decoded;
|
|
48
|
+
}
|
|
49
|
+
const dir = path.dirname(keyPath);
|
|
50
|
+
mkdirSync(dir, { recursive: true });
|
|
51
|
+
const generated = randomBytes(32);
|
|
52
|
+
writeFileSync(keyPath, generated.toString("base64"), { encoding: "utf8", mode: 0o600 });
|
|
53
|
+
try {
|
|
54
|
+
chmodSync(keyPath, 0o600);
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
// best effort
|
|
58
|
+
}
|
|
59
|
+
return generated;
|
|
60
|
+
}
|
|
61
|
+
function sha256Hex(value) {
|
|
62
|
+
return createHash("sha256").update(value).digest("hex");
|
|
63
|
+
}
|
|
64
|
+
function encryptValue(masterKey, value) {
|
|
65
|
+
const iv = randomBytes(12);
|
|
66
|
+
const cipher = createCipheriv("aes-256-gcm", masterKey, iv);
|
|
67
|
+
const ciphertext = Buffer.concat([cipher.update(value, "utf8"), cipher.final()]);
|
|
68
|
+
const tag = cipher.getAuthTag();
|
|
69
|
+
return {
|
|
70
|
+
scheme: "local_encrypted_v1",
|
|
71
|
+
iv: iv.toString("base64"),
|
|
72
|
+
tag: tag.toString("base64"),
|
|
73
|
+
ciphertext: ciphertext.toString("base64"),
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
function decryptValue(masterKey, material) {
|
|
77
|
+
const iv = Buffer.from(material.iv, "base64");
|
|
78
|
+
const tag = Buffer.from(material.tag, "base64");
|
|
79
|
+
const ciphertext = Buffer.from(material.ciphertext, "base64");
|
|
80
|
+
const decipher = createDecipheriv("aes-256-gcm", masterKey, iv);
|
|
81
|
+
decipher.setAuthTag(tag);
|
|
82
|
+
const plain = Buffer.concat([decipher.update(ciphertext), decipher.final()]);
|
|
83
|
+
return plain.toString("utf8");
|
|
84
|
+
}
|
|
85
|
+
function asLocalEncryptedMaterial(value) {
|
|
86
|
+
if (value &&
|
|
87
|
+
typeof value === "object" &&
|
|
88
|
+
value.scheme === "local_encrypted_v1" &&
|
|
89
|
+
typeof value.iv === "string" &&
|
|
90
|
+
typeof value.tag === "string" &&
|
|
91
|
+
typeof value.ciphertext === "string") {
|
|
92
|
+
return value;
|
|
93
|
+
}
|
|
94
|
+
throw badRequest("Invalid local_encrypted secret material");
|
|
95
|
+
}
|
|
96
|
+
export const localEncryptedProvider = {
|
|
97
|
+
id: "local_encrypted",
|
|
98
|
+
descriptor: {
|
|
99
|
+
id: "local_encrypted",
|
|
100
|
+
label: "Local encrypted (default)",
|
|
101
|
+
requiresExternalRef: false,
|
|
102
|
+
},
|
|
103
|
+
async createVersion(input) {
|
|
104
|
+
const masterKey = loadOrCreateMasterKey();
|
|
105
|
+
return {
|
|
106
|
+
material: encryptValue(masterKey, input.value),
|
|
107
|
+
valueSha256: sha256Hex(input.value),
|
|
108
|
+
externalRef: null,
|
|
109
|
+
};
|
|
110
|
+
},
|
|
111
|
+
async resolveVersion(input) {
|
|
112
|
+
const masterKey = loadOrCreateMasterKey();
|
|
113
|
+
return decryptValue(masterKey, asLocalEncryptedMaterial(input.material));
|
|
114
|
+
},
|
|
115
|
+
};
|
|
116
|
+
//# sourceMappingURL=local-encrypted-provider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"local-encrypted-provider.js","sourceRoot":"","sources":["../../src/secrets/local-encrypted-provider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { SecretProvider, SecretProviderDescriptor } from "@paperclipai/shared";
|
|
2
|
+
import type { SecretProviderModule } from "./types.js";
|
|
3
|
+
export declare function getSecretProvider(id: SecretProvider): SecretProviderModule;
|
|
4
|
+
export declare function listSecretProviders(): SecretProviderDescriptor[];
|
|
5
|
+
//# sourceMappingURL=provider-registry.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"provider-registry.d.ts","sourceRoot":"","sources":["../../src/secrets/provider-registry.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,wBAAwB,EAAE,MAAM,qBAAqB,CAAC;AAOpF,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAC;AAcvD,wBAAgB,iBAAiB,CAAC,EAAE,EAAE,cAAc,GAAG,oBAAoB,CAI1E;AAED,wBAAgB,mBAAmB,IAAI,wBAAwB,EAAE,CAEhE"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { localEncryptedProvider } from "./local-encrypted-provider.js";
|
|
2
|
+
import { awsSecretsManagerProvider, gcpSecretManagerProvider, vaultProvider, } from "./external-stub-providers.js";
|
|
3
|
+
import { unprocessable } from "../errors.js";
|
|
4
|
+
const providers = [
|
|
5
|
+
localEncryptedProvider,
|
|
6
|
+
awsSecretsManagerProvider,
|
|
7
|
+
gcpSecretManagerProvider,
|
|
8
|
+
vaultProvider,
|
|
9
|
+
];
|
|
10
|
+
const providerById = new Map(providers.map((provider) => [provider.id, provider]));
|
|
11
|
+
export function getSecretProvider(id) {
|
|
12
|
+
const provider = providerById.get(id);
|
|
13
|
+
if (!provider)
|
|
14
|
+
throw unprocessable(`Unsupported secret provider: ${id}`);
|
|
15
|
+
return provider;
|
|
16
|
+
}
|
|
17
|
+
export function listSecretProviders() {
|
|
18
|
+
return providers.map((provider) => provider.descriptor);
|
|
19
|
+
}
|
|
20
|
+
//# sourceMappingURL=provider-registry.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"provider-registry.js","sourceRoot":"","sources":["../../src/secrets/provider-registry.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,sBAAsB,EAAE,MAAM,+BAA+B,CAAC;AACvE,OAAO,EACL,yBAAyB,EACzB,wBAAwB,EACxB,aAAa,GACd,MAAM,8BAA8B,CAAC;AAEtC,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAE7C,MAAM,SAAS,GAA2B;IACxC,sBAAsB;IACtB,yBAAyB;IACzB,wBAAwB;IACxB,aAAa;CACd,CAAC;AAEF,MAAM,YAAY,GAAG,IAAI,GAAG,CAC1B,SAAS,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,QAAQ,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC,CACrD,CAAC;AAEF,MAAM,UAAU,iBAAiB,CAAC,EAAkB;IAClD,MAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IACtC,IAAI,CAAC,QAAQ;QAAE,MAAM,aAAa,CAAC,gCAAgC,EAAE,EAAE,CAAC,CAAC;IACzE,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,MAAM,UAAU,mBAAmB;IACjC,OAAO,SAAS,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;AAC1D,CAAC"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { SecretProvider, SecretProviderDescriptor } from "@paperclipai/shared";
|
|
2
|
+
export interface StoredSecretVersionMaterial {
|
|
3
|
+
[key: string]: unknown;
|
|
4
|
+
}
|
|
5
|
+
export interface SecretProviderModule {
|
|
6
|
+
id: SecretProvider;
|
|
7
|
+
descriptor: SecretProviderDescriptor;
|
|
8
|
+
createVersion(input: {
|
|
9
|
+
value: string;
|
|
10
|
+
externalRef: string | null;
|
|
11
|
+
}): Promise<{
|
|
12
|
+
material: StoredSecretVersionMaterial;
|
|
13
|
+
valueSha256: string;
|
|
14
|
+
externalRef: string | null;
|
|
15
|
+
}>;
|
|
16
|
+
resolveVersion(input: {
|
|
17
|
+
material: StoredSecretVersionMaterial;
|
|
18
|
+
externalRef: string | null;
|
|
19
|
+
}): Promise<string>;
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/secrets/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,wBAAwB,EAAE,MAAM,qBAAqB,CAAC;AAEpF,MAAM,WAAW,2BAA2B;IAC1C,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,oBAAoB;IACnC,EAAE,EAAE,cAAc,CAAC;IACnB,UAAU,EAAE,wBAAwB,CAAC;IACrC,aAAa,CAAC,KAAK,EAAE;QACnB,KAAK,EAAE,MAAM,CAAC;QACd,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;KAC5B,GAAG,OAAO,CAAC;QACV,QAAQ,EAAE,2BAA2B,CAAC;QACtC,WAAW,EAAE,MAAM,CAAC;QACpB,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;KAC5B,CAAC,CAAC;IACH,cAAc,CAAC,KAAK,EAAE;QACpB,QAAQ,EAAE,2BAA2B,CAAC;QACtC,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;KAC5B,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;CACrB"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/secrets/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import type { Db } from "@paperclipai/db";
|
|
2
|
+
import { companyMemberships } from "@paperclipai/db";
|
|
3
|
+
import type { PermissionKey, PrincipalType } from "@paperclipai/shared";
|
|
4
|
+
type MembershipRow = typeof companyMemberships.$inferSelect;
|
|
5
|
+
type GrantInput = {
|
|
6
|
+
permissionKey: PermissionKey;
|
|
7
|
+
scope?: Record<string, unknown> | null;
|
|
8
|
+
};
|
|
9
|
+
export declare function accessService(db: Db): {
|
|
10
|
+
isInstanceAdmin: (userId: string | null | undefined) => Promise<boolean>;
|
|
11
|
+
canUser: (companyId: string, userId: string | null | undefined, permissionKey: PermissionKey) => Promise<boolean>;
|
|
12
|
+
hasPermission: (companyId: string, principalType: PrincipalType, principalId: string, permissionKey: PermissionKey) => Promise<boolean>;
|
|
13
|
+
getMembership: (companyId: string, principalType: PrincipalType, principalId: string) => Promise<MembershipRow | null>;
|
|
14
|
+
ensureMembership: (companyId: string, principalType: PrincipalType, principalId: string, membershipRole?: string | null, status?: "pending" | "active" | "suspended") => Promise<{
|
|
15
|
+
id: string;
|
|
16
|
+
status: string;
|
|
17
|
+
createdAt: Date;
|
|
18
|
+
updatedAt: Date;
|
|
19
|
+
companyId: string;
|
|
20
|
+
principalType: string;
|
|
21
|
+
principalId: string;
|
|
22
|
+
membershipRole: string | null;
|
|
23
|
+
}>;
|
|
24
|
+
listMembers: (companyId: string) => Promise<{
|
|
25
|
+
id: string;
|
|
26
|
+
status: string;
|
|
27
|
+
createdAt: Date;
|
|
28
|
+
updatedAt: Date;
|
|
29
|
+
companyId: string;
|
|
30
|
+
principalType: string;
|
|
31
|
+
principalId: string;
|
|
32
|
+
membershipRole: string | null;
|
|
33
|
+
}[]>;
|
|
34
|
+
setMemberPermissions: (companyId: string, memberId: string, grants: GrantInput[], grantedByUserId: string | null) => Promise<{
|
|
35
|
+
id: string;
|
|
36
|
+
status: string;
|
|
37
|
+
createdAt: Date;
|
|
38
|
+
updatedAt: Date;
|
|
39
|
+
companyId: string;
|
|
40
|
+
principalType: string;
|
|
41
|
+
principalId: string;
|
|
42
|
+
membershipRole: string | null;
|
|
43
|
+
} | null>;
|
|
44
|
+
promoteInstanceAdmin: (userId: string) => Promise<{
|
|
45
|
+
id: string;
|
|
46
|
+
createdAt: Date;
|
|
47
|
+
updatedAt: Date;
|
|
48
|
+
userId: string;
|
|
49
|
+
role: string;
|
|
50
|
+
}>;
|
|
51
|
+
demoteInstanceAdmin: (userId: string) => Promise<{
|
|
52
|
+
id: string;
|
|
53
|
+
createdAt: Date;
|
|
54
|
+
updatedAt: Date;
|
|
55
|
+
userId: string;
|
|
56
|
+
role: string;
|
|
57
|
+
}>;
|
|
58
|
+
listUserCompanyAccess: (userId: string) => Promise<{
|
|
59
|
+
id: string;
|
|
60
|
+
status: string;
|
|
61
|
+
createdAt: Date;
|
|
62
|
+
updatedAt: Date;
|
|
63
|
+
companyId: string;
|
|
64
|
+
principalType: string;
|
|
65
|
+
principalId: string;
|
|
66
|
+
membershipRole: string | null;
|
|
67
|
+
}[]>;
|
|
68
|
+
setUserCompanyAccess: (userId: string, companyIds: string[]) => Promise<{
|
|
69
|
+
id: string;
|
|
70
|
+
status: string;
|
|
71
|
+
createdAt: Date;
|
|
72
|
+
updatedAt: Date;
|
|
73
|
+
companyId: string;
|
|
74
|
+
principalType: string;
|
|
75
|
+
principalId: string;
|
|
76
|
+
membershipRole: string | null;
|
|
77
|
+
}[]>;
|
|
78
|
+
setPrincipalGrants: (companyId: string, principalType: PrincipalType, principalId: string, grants: GrantInput[], grantedByUserId: string | null) => Promise<void>;
|
|
79
|
+
};
|
|
80
|
+
export {};
|
|
81
|
+
//# sourceMappingURL=access.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"access.d.ts","sourceRoot":"","sources":["../../src/services/access.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,EAAE,EAAE,MAAM,iBAAiB,CAAC;AAC1C,OAAO,EACL,kBAAkB,EAGnB,MAAM,iBAAiB,CAAC;AACzB,OAAO,KAAK,EAAE,aAAa,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAExE,KAAK,aAAa,GAAG,OAAO,kBAAkB,CAAC,YAAY,CAAC;AAC5D,KAAK,UAAU,GAAG;IAChB,aAAa,EAAE,aAAa,CAAC;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;CACxC,CAAC;AAEF,wBAAgB,aAAa,CAAC,EAAE,EAAE,EAAE;8BACK,MAAM,GAAG,IAAI,GAAG,SAAS,KAAG,OAAO,CAAC,OAAO,CAAC;yBAoDtE,MAAM,UACT,MAAM,GAAG,IAAI,GAAG,SAAS,iBAClB,aAAa,KAC3B,OAAO,CAAC,OAAO,CAAC;+BA1BN,MAAM,iBACF,aAAa,eACf,MAAM,iBACJ,aAAa,KAC3B,OAAO,CAAC,OAAO,CAAC;+BAtBN,MAAM,iBACF,aAAa,eACf,MAAM,KAClB,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC;kCA6JnB,MAAM,iBACF,aAAa,eACf,MAAM,mBACH,MAAM,GAAG,IAAI,WACrB,SAAS,GAAG,QAAQ,GAAG,WAAW;;;;;;;;;;6BAlHN,MAAM;;;;;;;;;;sCAS/B,MAAM,YACP,MAAM,UACR,UAAU,EAAE,mBACH,MAAM,GAAG,IAAI;;;;;;;;;;mCAsCY,MAAM;;;;;;;kCAiBP,MAAM;;;;;;;oCAQJ,MAAM;;;;;;;;;;mCAQP,MAAM,cAAc,MAAM,EAAE;;;;;;;;;;oCA6D3D,MAAM,iBACF,aAAa,eACf,MAAM,UACX,UAAU,EAAE,mBACH,MAAM,GAAG,IAAI;EA0CjC"}
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
import { and, eq, inArray, sql } from "drizzle-orm";
|
|
2
|
+
import { companyMemberships, instanceUserRoles, principalPermissionGrants, } from "@paperclipai/db";
|
|
3
|
+
export function accessService(db) {
|
|
4
|
+
async function isInstanceAdmin(userId) {
|
|
5
|
+
if (!userId)
|
|
6
|
+
return false;
|
|
7
|
+
const row = await db
|
|
8
|
+
.select({ id: instanceUserRoles.id })
|
|
9
|
+
.from(instanceUserRoles)
|
|
10
|
+
.where(and(eq(instanceUserRoles.userId, userId), eq(instanceUserRoles.role, "instance_admin")))
|
|
11
|
+
.then((rows) => rows[0] ?? null);
|
|
12
|
+
return Boolean(row);
|
|
13
|
+
}
|
|
14
|
+
async function getMembership(companyId, principalType, principalId) {
|
|
15
|
+
return db
|
|
16
|
+
.select()
|
|
17
|
+
.from(companyMemberships)
|
|
18
|
+
.where(and(eq(companyMemberships.companyId, companyId), eq(companyMemberships.principalType, principalType), eq(companyMemberships.principalId, principalId)))
|
|
19
|
+
.then((rows) => rows[0] ?? null);
|
|
20
|
+
}
|
|
21
|
+
async function hasPermission(companyId, principalType, principalId, permissionKey) {
|
|
22
|
+
const membership = await getMembership(companyId, principalType, principalId);
|
|
23
|
+
if (!membership || membership.status !== "active")
|
|
24
|
+
return false;
|
|
25
|
+
const grant = await db
|
|
26
|
+
.select({ id: principalPermissionGrants.id })
|
|
27
|
+
.from(principalPermissionGrants)
|
|
28
|
+
.where(and(eq(principalPermissionGrants.companyId, companyId), eq(principalPermissionGrants.principalType, principalType), eq(principalPermissionGrants.principalId, principalId), eq(principalPermissionGrants.permissionKey, permissionKey)))
|
|
29
|
+
.then((rows) => rows[0] ?? null);
|
|
30
|
+
return Boolean(grant);
|
|
31
|
+
}
|
|
32
|
+
async function canUser(companyId, userId, permissionKey) {
|
|
33
|
+
if (!userId)
|
|
34
|
+
return false;
|
|
35
|
+
if (await isInstanceAdmin(userId))
|
|
36
|
+
return true;
|
|
37
|
+
return hasPermission(companyId, "user", userId, permissionKey);
|
|
38
|
+
}
|
|
39
|
+
async function listMembers(companyId) {
|
|
40
|
+
return db
|
|
41
|
+
.select()
|
|
42
|
+
.from(companyMemberships)
|
|
43
|
+
.where(eq(companyMemberships.companyId, companyId))
|
|
44
|
+
.orderBy(sql `${companyMemberships.createdAt} desc`);
|
|
45
|
+
}
|
|
46
|
+
async function setMemberPermissions(companyId, memberId, grants, grantedByUserId) {
|
|
47
|
+
const member = await db
|
|
48
|
+
.select()
|
|
49
|
+
.from(companyMemberships)
|
|
50
|
+
.where(and(eq(companyMemberships.companyId, companyId), eq(companyMemberships.id, memberId)))
|
|
51
|
+
.then((rows) => rows[0] ?? null);
|
|
52
|
+
if (!member)
|
|
53
|
+
return null;
|
|
54
|
+
await db.transaction(async (tx) => {
|
|
55
|
+
await tx
|
|
56
|
+
.delete(principalPermissionGrants)
|
|
57
|
+
.where(and(eq(principalPermissionGrants.companyId, companyId), eq(principalPermissionGrants.principalType, member.principalType), eq(principalPermissionGrants.principalId, member.principalId)));
|
|
58
|
+
if (grants.length > 0) {
|
|
59
|
+
await tx.insert(principalPermissionGrants).values(grants.map((grant) => ({
|
|
60
|
+
companyId,
|
|
61
|
+
principalType: member.principalType,
|
|
62
|
+
principalId: member.principalId,
|
|
63
|
+
permissionKey: grant.permissionKey,
|
|
64
|
+
scope: grant.scope ?? null,
|
|
65
|
+
grantedByUserId,
|
|
66
|
+
createdAt: new Date(),
|
|
67
|
+
updatedAt: new Date(),
|
|
68
|
+
})));
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
return member;
|
|
72
|
+
}
|
|
73
|
+
async function promoteInstanceAdmin(userId) {
|
|
74
|
+
const existing = await db
|
|
75
|
+
.select()
|
|
76
|
+
.from(instanceUserRoles)
|
|
77
|
+
.where(and(eq(instanceUserRoles.userId, userId), eq(instanceUserRoles.role, "instance_admin")))
|
|
78
|
+
.then((rows) => rows[0] ?? null);
|
|
79
|
+
if (existing)
|
|
80
|
+
return existing;
|
|
81
|
+
return db
|
|
82
|
+
.insert(instanceUserRoles)
|
|
83
|
+
.values({
|
|
84
|
+
userId,
|
|
85
|
+
role: "instance_admin",
|
|
86
|
+
})
|
|
87
|
+
.returning()
|
|
88
|
+
.then((rows) => rows[0]);
|
|
89
|
+
}
|
|
90
|
+
async function demoteInstanceAdmin(userId) {
|
|
91
|
+
return db
|
|
92
|
+
.delete(instanceUserRoles)
|
|
93
|
+
.where(and(eq(instanceUserRoles.userId, userId), eq(instanceUserRoles.role, "instance_admin")))
|
|
94
|
+
.returning()
|
|
95
|
+
.then((rows) => rows[0] ?? null);
|
|
96
|
+
}
|
|
97
|
+
async function listUserCompanyAccess(userId) {
|
|
98
|
+
return db
|
|
99
|
+
.select()
|
|
100
|
+
.from(companyMemberships)
|
|
101
|
+
.where(and(eq(companyMemberships.principalType, "user"), eq(companyMemberships.principalId, userId)))
|
|
102
|
+
.orderBy(sql `${companyMemberships.createdAt} desc`);
|
|
103
|
+
}
|
|
104
|
+
async function setUserCompanyAccess(userId, companyIds) {
|
|
105
|
+
const existing = await listUserCompanyAccess(userId);
|
|
106
|
+
const existingByCompany = new Map(existing.map((row) => [row.companyId, row]));
|
|
107
|
+
const target = new Set(companyIds);
|
|
108
|
+
await db.transaction(async (tx) => {
|
|
109
|
+
const toDelete = existing.filter((row) => !target.has(row.companyId)).map((row) => row.id);
|
|
110
|
+
if (toDelete.length > 0) {
|
|
111
|
+
await tx.delete(companyMemberships).where(inArray(companyMemberships.id, toDelete));
|
|
112
|
+
}
|
|
113
|
+
for (const companyId of target) {
|
|
114
|
+
if (existingByCompany.has(companyId))
|
|
115
|
+
continue;
|
|
116
|
+
await tx.insert(companyMemberships).values({
|
|
117
|
+
companyId,
|
|
118
|
+
principalType: "user",
|
|
119
|
+
principalId: userId,
|
|
120
|
+
status: "active",
|
|
121
|
+
membershipRole: "member",
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
return listUserCompanyAccess(userId);
|
|
126
|
+
}
|
|
127
|
+
async function ensureMembership(companyId, principalType, principalId, membershipRole = "member", status = "active") {
|
|
128
|
+
const existing = await getMembership(companyId, principalType, principalId);
|
|
129
|
+
if (existing) {
|
|
130
|
+
if (existing.status !== status || existing.membershipRole !== membershipRole) {
|
|
131
|
+
const updated = await db
|
|
132
|
+
.update(companyMemberships)
|
|
133
|
+
.set({ status, membershipRole, updatedAt: new Date() })
|
|
134
|
+
.where(eq(companyMemberships.id, existing.id))
|
|
135
|
+
.returning()
|
|
136
|
+
.then((rows) => rows[0] ?? null);
|
|
137
|
+
return updated ?? existing;
|
|
138
|
+
}
|
|
139
|
+
return existing;
|
|
140
|
+
}
|
|
141
|
+
return db
|
|
142
|
+
.insert(companyMemberships)
|
|
143
|
+
.values({
|
|
144
|
+
companyId,
|
|
145
|
+
principalType,
|
|
146
|
+
principalId,
|
|
147
|
+
status,
|
|
148
|
+
membershipRole,
|
|
149
|
+
})
|
|
150
|
+
.returning()
|
|
151
|
+
.then((rows) => rows[0]);
|
|
152
|
+
}
|
|
153
|
+
async function setPrincipalGrants(companyId, principalType, principalId, grants, grantedByUserId) {
|
|
154
|
+
await db.transaction(async (tx) => {
|
|
155
|
+
await tx
|
|
156
|
+
.delete(principalPermissionGrants)
|
|
157
|
+
.where(and(eq(principalPermissionGrants.companyId, companyId), eq(principalPermissionGrants.principalType, principalType), eq(principalPermissionGrants.principalId, principalId)));
|
|
158
|
+
if (grants.length === 0)
|
|
159
|
+
return;
|
|
160
|
+
await tx.insert(principalPermissionGrants).values(grants.map((grant) => ({
|
|
161
|
+
companyId,
|
|
162
|
+
principalType,
|
|
163
|
+
principalId,
|
|
164
|
+
permissionKey: grant.permissionKey,
|
|
165
|
+
scope: grant.scope ?? null,
|
|
166
|
+
grantedByUserId,
|
|
167
|
+
createdAt: new Date(),
|
|
168
|
+
updatedAt: new Date(),
|
|
169
|
+
})));
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
return {
|
|
173
|
+
isInstanceAdmin,
|
|
174
|
+
canUser,
|
|
175
|
+
hasPermission,
|
|
176
|
+
getMembership,
|
|
177
|
+
ensureMembership,
|
|
178
|
+
listMembers,
|
|
179
|
+
setMemberPermissions,
|
|
180
|
+
promoteInstanceAdmin,
|
|
181
|
+
demoteInstanceAdmin,
|
|
182
|
+
listUserCompanyAccess,
|
|
183
|
+
setUserCompanyAccess,
|
|
184
|
+
setPrincipalGrants,
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
//# sourceMappingURL=access.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"access.js","sourceRoot":"","sources":["../../src/services/access.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { Db } from "@paperclipai/db";
|
|
2
|
+
export interface LogActivityInput {
|
|
3
|
+
companyId: string;
|
|
4
|
+
actorType: "agent" | "user" | "system";
|
|
5
|
+
actorId: string;
|
|
6
|
+
action: string;
|
|
7
|
+
entityType: string;
|
|
8
|
+
entityId: string;
|
|
9
|
+
agentId?: string | null;
|
|
10
|
+
runId?: string | null;
|
|
11
|
+
details?: Record<string, unknown> | null;
|
|
12
|
+
}
|
|
13
|
+
export declare function logActivity(db: Db, input: LogActivityInput): Promise<void>;
|
|
14
|
+
//# sourceMappingURL=activity-log.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"activity-log.d.ts","sourceRoot":"","sources":["../../src/services/activity-log.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,EAAE,MAAM,iBAAiB,CAAC;AAK1C,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,OAAO,GAAG,MAAM,GAAG,QAAQ,CAAC;IACvC,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAC;CAC1C;AAED,wBAAsB,WAAW,CAAC,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,gBAAgB,iBA4BhE"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { activityLog } from "@paperclipai/db";
|
|
2
|
+
import { publishLiveEvent } from "./live-events.js";
|
|
3
|
+
import { sanitizeRecord } from "../redaction.js";
|
|
4
|
+
export async function logActivity(db, input) {
|
|
5
|
+
const sanitizedDetails = input.details ? sanitizeRecord(input.details) : null;
|
|
6
|
+
await db.insert(activityLog).values({
|
|
7
|
+
companyId: input.companyId,
|
|
8
|
+
actorType: input.actorType,
|
|
9
|
+
actorId: input.actorId,
|
|
10
|
+
action: input.action,
|
|
11
|
+
entityType: input.entityType,
|
|
12
|
+
entityId: input.entityId,
|
|
13
|
+
agentId: input.agentId ?? null,
|
|
14
|
+
runId: input.runId ?? null,
|
|
15
|
+
details: sanitizedDetails,
|
|
16
|
+
});
|
|
17
|
+
publishLiveEvent({
|
|
18
|
+
companyId: input.companyId,
|
|
19
|
+
type: "activity.logged",
|
|
20
|
+
payload: {
|
|
21
|
+
actorType: input.actorType,
|
|
22
|
+
actorId: input.actorId,
|
|
23
|
+
action: input.action,
|
|
24
|
+
entityType: input.entityType,
|
|
25
|
+
entityId: input.entityId,
|
|
26
|
+
agentId: input.agentId ?? null,
|
|
27
|
+
runId: input.runId ?? null,
|
|
28
|
+
details: sanitizedDetails,
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=activity-log.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"activity-log.js","sourceRoot":"","sources":["../../src/services/activity-log.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAcjD,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,EAAM,EAAE,KAAuB;IAC/D,MAAM,gBAAgB,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,cAAc,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC9E,MAAM,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC;QAClC,SAAS,EAAE,KAAK,CAAC,SAAS;QAC1B,SAAS,EAAE,KAAK,CAAC,SAAS;QAC1B,OAAO,EAAE,KAAK,CAAC,OAAO;QACtB,MAAM,EAAE,KAAK,CAAC,MAAM;QACpB,UAAU,EAAE,KAAK,CAAC,UAAU;QAC5B,QAAQ,EAAE,KAAK,CAAC,QAAQ;QACxB,OAAO,EAAE,KAAK,CAAC,OAAO,IAAI,IAAI;QAC9B,KAAK,EAAE,KAAK,CAAC,KAAK,IAAI,IAAI;QAC1B,OAAO,EAAE,gBAAgB;KAC1B,CAAC,CAAC;IAEH,gBAAgB,CAAC;QACf,SAAS,EAAE,KAAK,CAAC,SAAS;QAC1B,IAAI,EAAE,iBAAiB;QACvB,OAAO,EAAE;YACP,SAAS,EAAE,KAAK,CAAC,SAAS;YAC1B,OAAO,EAAE,KAAK,CAAC,OAAO;YACtB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,OAAO,EAAE,KAAK,CAAC,OAAO,IAAI,IAAI;YAC9B,KAAK,EAAE,KAAK,CAAC,KAAK,IAAI,IAAI;YAC1B,OAAO,EAAE,gBAAgB;SAC1B;KACF,CAAC,CAAC;AACL,CAAC"}
|