botmux 2.44.0 → 2.46.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.en.md +1 -2
- package/README.md +1 -2
- package/dist/cli/bots-list-output.d.ts +8 -0
- package/dist/cli/bots-list-output.d.ts.map +1 -1
- package/dist/cli/bots-list-output.js +9 -0
- package/dist/cli/bots-list-output.js.map +1 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +6 -0
- package/dist/cli.js.map +1 -1
- package/dist/core/command-handler.d.ts.map +1 -1
- package/dist/core/command-handler.js +601 -50
- package/dist/core/command-handler.js.map +1 -1
- package/dist/core/dashboard-ipc-server.d.ts +2 -0
- package/dist/core/dashboard-ipc-server.d.ts.map +1 -1
- package/dist/core/dashboard-ipc-server.js +170 -2
- package/dist/core/dashboard-ipc-server.js.map +1 -1
- package/dist/core/role-resolver.d.ts +17 -1
- package/dist/core/role-resolver.d.ts.map +1 -1
- package/dist/core/role-resolver.js +64 -10
- package/dist/core/role-resolver.js.map +1 -1
- package/dist/core/session-manager.d.ts +1 -1
- package/dist/core/session-manager.d.ts.map +1 -1
- package/dist/core/session-manager.js +24 -13
- package/dist/core/session-manager.js.map +1 -1
- package/dist/core/trigger-session.d.ts +9 -0
- package/dist/core/trigger-session.d.ts.map +1 -0
- package/dist/core/trigger-session.js +158 -0
- package/dist/core/trigger-session.js.map +1 -0
- package/dist/core/worker-pool.d.ts +105 -0
- package/dist/core/worker-pool.d.ts.map +1 -1
- package/dist/core/worker-pool.js +284 -4
- package/dist/core/worker-pool.js.map +1 -1
- package/dist/daemon.d.ts.map +1 -1
- package/dist/daemon.js +80 -45
- package/dist/daemon.js.map +1 -1
- package/dist/dashboard/connector-api.d.ts +3 -0
- package/dist/dashboard/connector-api.d.ts.map +1 -0
- package/dist/dashboard/connector-api.js +351 -0
- package/dist/dashboard/connector-api.js.map +1 -0
- package/dist/dashboard/federated-group-core.d.ts +54 -0
- package/dist/dashboard/federated-group-core.d.ts.map +1 -0
- package/dist/dashboard/federated-group-core.js +165 -0
- package/dist/dashboard/federated-group-core.js.map +1 -0
- package/dist/dashboard/federation-api.d.ts +42 -0
- package/dist/dashboard/federation-api.d.ts.map +1 -0
- package/dist/dashboard/federation-api.js +408 -0
- package/dist/dashboard/federation-api.js.map +1 -0
- package/dist/dashboard/federation-spoke-api.d.ts +76 -0
- package/dist/dashboard/federation-spoke-api.d.ts.map +1 -0
- package/dist/dashboard/federation-spoke-api.js +618 -0
- package/dist/dashboard/federation-spoke-api.js.map +1 -0
- package/dist/dashboard/team-group.d.ts +18 -0
- package/dist/dashboard/team-group.d.ts.map +1 -0
- package/dist/dashboard/team-group.js +7 -0
- package/dist/dashboard/team-group.js.map +1 -0
- package/dist/dashboard/trigger-api.d.ts +13 -0
- package/dist/dashboard/trigger-api.d.ts.map +1 -0
- package/dist/dashboard/trigger-api.js +77 -0
- package/dist/dashboard/trigger-api.js.map +1 -0
- package/dist/dashboard/web/app.js +8 -0
- package/dist/dashboard/web/app.js.map +1 -1
- package/dist/dashboard/web/connectors.d.ts +2 -0
- package/dist/dashboard/web/connectors.d.ts.map +1 -0
- package/dist/dashboard/web/connectors.js +187 -0
- package/dist/dashboard/web/connectors.js.map +1 -0
- package/dist/dashboard/web/team-federation.d.ts +3 -0
- package/dist/dashboard/web/team-federation.d.ts.map +1 -0
- package/dist/dashboard/web/team-federation.js +487 -0
- package/dist/dashboard/web/team-federation.js.map +1 -0
- package/dist/dashboard/webhook-routes.d.ts +19 -0
- package/dist/dashboard/webhook-routes.d.ts.map +1 -0
- package/dist/dashboard/webhook-routes.js +321 -0
- package/dist/dashboard/webhook-routes.js.map +1 -0
- package/dist/dashboard-web/app.js +509 -386
- package/dist/dashboard-web/index.html +2 -0
- package/dist/dashboard.js +152 -1
- package/dist/dashboard.js.map +1 -1
- package/dist/i18n/en.d.ts.map +1 -1
- package/dist/i18n/en.js +85 -8
- package/dist/i18n/en.js.map +1 -1
- package/dist/i18n/zh.d.ts.map +1 -1
- package/dist/i18n/zh.js +85 -8
- package/dist/i18n/zh.js.map +1 -1
- package/dist/im/lark/card-builder.d.ts +82 -0
- package/dist/im/lark/card-builder.d.ts.map +1 -1
- package/dist/im/lark/card-builder.js +315 -0
- package/dist/im/lark/card-builder.js.map +1 -1
- package/dist/im/lark/card-handler.d.ts.map +1 -1
- package/dist/im/lark/card-handler.js +195 -1
- package/dist/im/lark/card-handler.js.map +1 -1
- package/dist/im/lark/client.d.ts +45 -0
- package/dist/im/lark/client.d.ts.map +1 -1
- package/dist/im/lark/client.js +169 -18
- package/dist/im/lark/client.js.map +1 -1
- package/dist/im/lark/message-parser.d.ts.map +1 -1
- package/dist/im/lark/message-parser.js +1 -0
- package/dist/im/lark/message-parser.js.map +1 -1
- package/dist/services/bot-owner-store.d.ts +28 -0
- package/dist/services/bot-owner-store.d.ts.map +1 -0
- package/dist/services/bot-owner-store.js +82 -0
- package/dist/services/bot-owner-store.js.map +1 -0
- package/dist/services/bot-profile-store.d.ts +16 -0
- package/dist/services/bot-profile-store.d.ts.map +1 -0
- package/dist/services/bot-profile-store.js +98 -0
- package/dist/services/bot-profile-store.js.map +1 -0
- package/dist/services/connector-store.d.ts +58 -0
- package/dist/services/connector-store.d.ts.map +1 -0
- package/dist/services/connector-store.js +79 -0
- package/dist/services/connector-store.js.map +1 -0
- package/dist/services/deployment-identity.d.ts +22 -0
- package/dist/services/deployment-identity.d.ts.map +1 -0
- package/dist/services/deployment-identity.js +67 -0
- package/dist/services/deployment-identity.js.map +1 -0
- package/dist/services/federation-membership-store.d.ts +23 -0
- package/dist/services/federation-membership-store.d.ts.map +1 -0
- package/dist/services/federation-membership-store.js +66 -0
- package/dist/services/federation-membership-store.js.map +1 -0
- package/dist/services/federation-roster.d.ts +54 -0
- package/dist/services/federation-roster.d.ts.map +1 -0
- package/dist/services/federation-roster.js +51 -0
- package/dist/services/federation-roster.js.map +1 -0
- package/dist/services/federation-store.d.ts +76 -0
- package/dist/services/federation-store.d.ts.map +1 -0
- package/dist/services/federation-store.js +133 -0
- package/dist/services/federation-store.js.map +1 -0
- package/dist/services/group-creator.d.ts +6 -0
- package/dist/services/group-creator.d.ts.map +1 -1
- package/dist/services/group-creator.js +10 -1
- package/dist/services/group-creator.js.map +1 -1
- package/dist/services/groups-store.d.ts +11 -0
- package/dist/services/groups-store.d.ts.map +1 -1
- package/dist/services/groups-store.js +42 -0
- package/dist/services/groups-store.js.map +1 -1
- package/dist/services/invite-store.d.ts +28 -0
- package/dist/services/invite-store.d.ts.map +1 -0
- package/dist/services/invite-store.js +85 -0
- package/dist/services/invite-store.js.map +1 -0
- package/dist/services/pairing-store.d.ts +47 -0
- package/dist/services/pairing-store.d.ts.map +1 -0
- package/dist/services/pairing-store.js +132 -0
- package/dist/services/pairing-store.js.map +1 -0
- package/dist/services/relay-picker.d.ts +22 -0
- package/dist/services/relay-picker.d.ts.map +1 -0
- package/dist/services/relay-picker.js +62 -0
- package/dist/services/relay-picker.js.map +1 -0
- package/dist/services/team-roster.d.ts +38 -0
- package/dist/services/team-roster.d.ts.map +1 -0
- package/dist/services/team-roster.js +82 -0
- package/dist/services/team-roster.js.map +1 -0
- package/dist/services/team-store.d.ts +54 -0
- package/dist/services/team-store.d.ts.map +1 -0
- package/dist/services/team-store.js +156 -0
- package/dist/services/team-store.js.map +1 -0
- package/dist/services/trigger-log-store.d.ts +46 -0
- package/dist/services/trigger-log-store.d.ts.map +1 -0
- package/dist/services/trigger-log-store.js +132 -0
- package/dist/services/trigger-log-store.js.map +1 -0
- package/dist/services/trigger-types.d.ts +57 -0
- package/dist/services/trigger-types.d.ts.map +1 -0
- package/dist/services/trigger-types.js +28 -0
- package/dist/services/trigger-types.js.map +1 -0
- package/dist/services/webhook-key.d.ts +16 -0
- package/dist/services/webhook-key.d.ts.map +1 -0
- package/dist/services/webhook-key.js +123 -0
- package/dist/services/webhook-key.js.map +1 -0
- package/dist/services/webhook-lifecycle-extractors.d.ts +15 -0
- package/dist/services/webhook-lifecycle-extractors.d.ts.map +1 -0
- package/dist/services/webhook-lifecycle-extractors.js +59 -0
- package/dist/services/webhook-lifecycle-extractors.js.map +1 -0
- package/dist/services/webhook-lifecycle-store.d.ts +45 -0
- package/dist/services/webhook-lifecycle-store.d.ts.map +1 -0
- package/dist/services/webhook-lifecycle-store.js +159 -0
- package/dist/services/webhook-lifecycle-store.js.map +1 -0
- package/dist/setup/verify-permissions.d.ts.map +1 -1
- package/dist/setup/verify-permissions.js +4 -0
- package/dist/setup/verify-permissions.js.map +1 -1
- package/dist/skills/definitions.d.ts.map +1 -1
- package/dist/skills/definitions.js +64 -10
- package/dist/skills/definitions.js.map +1 -1
- package/dist/types.d.ts +14 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/utils/daemon-discovery.d.ts +11 -0
- package/dist/utils/daemon-discovery.d.ts.map +1 -0
- package/dist/utils/daemon-discovery.js +59 -0
- package/dist/utils/daemon-discovery.js.map +1 -0
- package/dist/workflows/events/payloads.d.ts +2 -2
- package/dist/workflows/events/schema.d.ts +8 -8
- package/dist/workflows/trigger-from-envelope.d.ts +13 -0
- package/dist/workflows/trigger-from-envelope.d.ts.map +1 -0
- package/dist/workflows/trigger-from-envelope.js +67 -0
- package/dist/workflows/trigger-from-envelope.js.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Team invite store: an existing member mints a single-use, short-TTL invite
|
|
3
|
+
* code; the invitee pairs (pairing-login) and presents the code on consume,
|
|
4
|
+
* which authorizes joining the team (team-store.addMember). This sidesteps
|
|
5
|
+
* email→open_id resolution (the dashboard process has no Lark client) — the
|
|
6
|
+
* invitee self-authenticates via the bot, the invite only grants admission.
|
|
7
|
+
*
|
|
8
|
+
* Storage: `{dataDir}/team-invites.json`, atomic writes; expired/used pruned.
|
|
9
|
+
*
|
|
10
|
+
* Concurrency: consume is a read-modify-write. This assumes a SINGLE dashboard
|
|
11
|
+
* writer process (the current deployment model). If multiple dashboard processes
|
|
12
|
+
* ever write concurrently, a single-use code would need a file lock here.
|
|
13
|
+
*/
|
|
14
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync, renameSync } from 'node:fs';
|
|
15
|
+
import { join } from 'node:path';
|
|
16
|
+
import { randomBytes, randomUUID } from 'node:crypto';
|
|
17
|
+
const DEFAULT_TTL_MS = 24 * 60 * 60 * 1000; // 24h
|
|
18
|
+
function filePath(dataDir) { return join(dataDir, 'team-invites.json'); }
|
|
19
|
+
function readFile(dataDir) {
|
|
20
|
+
const fp = filePath(dataDir);
|
|
21
|
+
if (!existsSync(fp))
|
|
22
|
+
return {};
|
|
23
|
+
try {
|
|
24
|
+
const parsed = JSON.parse(readFileSync(fp, 'utf-8'));
|
|
25
|
+
if (parsed && typeof parsed === 'object' && !Array.isArray(parsed))
|
|
26
|
+
return parsed;
|
|
27
|
+
}
|
|
28
|
+
catch { /* corrupt */ }
|
|
29
|
+
return {};
|
|
30
|
+
}
|
|
31
|
+
function writeFileAtomic(dataDir, data) {
|
|
32
|
+
if (!existsSync(dataDir))
|
|
33
|
+
mkdirSync(dataDir, { recursive: true });
|
|
34
|
+
const fp = filePath(dataDir);
|
|
35
|
+
const tmp = `${fp}.${process.pid}.${randomUUID()}.tmp`;
|
|
36
|
+
writeFileSync(tmp, JSON.stringify(data, null, 2) + '\n', 'utf-8');
|
|
37
|
+
renameSync(tmp, fp);
|
|
38
|
+
}
|
|
39
|
+
/** Drop expired or used-and-old entries. */
|
|
40
|
+
function prune(data, now) {
|
|
41
|
+
for (const [code, inv] of Object.entries(data)) {
|
|
42
|
+
if (inv.expiresAt <= now || (inv.usedAt && now - inv.usedAt > DEFAULT_TTL_MS))
|
|
43
|
+
delete data[code];
|
|
44
|
+
}
|
|
45
|
+
return data;
|
|
46
|
+
}
|
|
47
|
+
/** Mint a single-use invite for a team. */
|
|
48
|
+
export function createInvite(dataDir, teamId, createdBy, ttlMs = DEFAULT_TTL_MS, now = Date.now()) {
|
|
49
|
+
const data = prune(readFile(dataDir), now);
|
|
50
|
+
const code = randomBytes(9).toString('base64url'); // ~12 chars, high entropy
|
|
51
|
+
data[code] = { code, teamId, createdBy, createdAt: now, expiresAt: now + ttlMs };
|
|
52
|
+
writeFileAtomic(dataDir, data);
|
|
53
|
+
return { code, expiresAt: now + ttlMs };
|
|
54
|
+
}
|
|
55
|
+
/** Drop every invite for a team (e.g. when the team is deleted), so stale codes
|
|
56
|
+
* can't later be consumed to join/switch into a non-existent team. Returns count. */
|
|
57
|
+
export function deleteInvitesForTeam(dataDir, teamId) {
|
|
58
|
+
const data = readFile(dataDir);
|
|
59
|
+
let removed = 0;
|
|
60
|
+
for (const [code, inv] of Object.entries(data)) {
|
|
61
|
+
if (inv.teamId === teamId) {
|
|
62
|
+
delete data[code];
|
|
63
|
+
removed++;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
if (removed)
|
|
67
|
+
writeFileAtomic(dataDir, data);
|
|
68
|
+
return removed;
|
|
69
|
+
}
|
|
70
|
+
/** Validate + burn an invite (single-use). Determines the precise failure
|
|
71
|
+
* reason BEFORE pruning, so an expired code reports `expired` (not `not_found`). */
|
|
72
|
+
export function consumeInvite(dataDir, code, now = Date.now()) {
|
|
73
|
+
const data = readFile(dataDir);
|
|
74
|
+
const inv = data[code?.trim()];
|
|
75
|
+
if (!inv)
|
|
76
|
+
return { ok: false, reason: 'not_found' };
|
|
77
|
+
if (inv.expiresAt <= now)
|
|
78
|
+
return { ok: false, reason: 'expired' };
|
|
79
|
+
if (inv.usedAt)
|
|
80
|
+
return { ok: false, reason: 'used' };
|
|
81
|
+
inv.usedAt = now;
|
|
82
|
+
writeFileAtomic(dataDir, prune(data, now)); // burn this one; prune other stale entries
|
|
83
|
+
return { ok: true, teamId: inv.teamId };
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=invite-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"invite-store.js","sourceRoot":"","sources":["../../src/services/invite-store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export type PairingStatus = 'pending' | 'claimed' | 'consumed';
|
|
2
|
+
export interface PairingClaimer {
|
|
3
|
+
openId: string;
|
|
4
|
+
unionId?: string;
|
|
5
|
+
name?: string;
|
|
6
|
+
/** The bot app the user ran `/pair` with — open_id is scoped to THIS app. */
|
|
7
|
+
larkAppId?: string;
|
|
8
|
+
}
|
|
9
|
+
export interface StartedPairing {
|
|
10
|
+
pairingId: string;
|
|
11
|
+
code: string;
|
|
12
|
+
browserToken: string;
|
|
13
|
+
expiresAt: number;
|
|
14
|
+
}
|
|
15
|
+
/** Begin a pairing. Returns the code to show the user + a private browserToken. */
|
|
16
|
+
export declare function createPairing(dataDir: string, ttlMs?: number, now?: number): StartedPairing;
|
|
17
|
+
export type ClaimResult = {
|
|
18
|
+
ok: true;
|
|
19
|
+
pairingId: string;
|
|
20
|
+
} | {
|
|
21
|
+
ok: false;
|
|
22
|
+
reason: 'not_found' | 'expired' | 'already_claimed';
|
|
23
|
+
};
|
|
24
|
+
/** Claim a pending pairing for a Feishu identity (called by the daemon on the bot side). */
|
|
25
|
+
export declare function claimPairing(dataDir: string, code: string, claimer: PairingClaimer, now?: number): ClaimResult;
|
|
26
|
+
export type PairingView = {
|
|
27
|
+
status: 'pending';
|
|
28
|
+
} | {
|
|
29
|
+
status: 'claimed';
|
|
30
|
+
claimedBy: PairingClaimer;
|
|
31
|
+
} | {
|
|
32
|
+
status: 'consumed';
|
|
33
|
+
} | {
|
|
34
|
+
status: 'not_found';
|
|
35
|
+
};
|
|
36
|
+
/** Browser-side status poll; requires the matching browserToken. */
|
|
37
|
+
export declare function getPairingStatus(dataDir: string, pairingId: string, browserToken: string, now?: number): PairingView;
|
|
38
|
+
export type ConsumeResult = {
|
|
39
|
+
ok: true;
|
|
40
|
+
claimedBy: PairingClaimer;
|
|
41
|
+
} | {
|
|
42
|
+
ok: false;
|
|
43
|
+
reason: 'not_found' | 'not_claimed' | 'already_consumed';
|
|
44
|
+
};
|
|
45
|
+
/** Single-use: turn a claimed pairing into a session. Requires the browserToken. */
|
|
46
|
+
export declare function consumePairing(dataDir: string, pairingId: string, browserToken: string, now?: number): ConsumeResult;
|
|
47
|
+
//# sourceMappingURL=pairing-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pairing-store.d.ts","sourceRoot":"","sources":["../../src/services/pairing-store.ts"],"names":[],"mappings":"AAiCA,MAAM,MAAM,aAAa,GAAG,SAAS,GAAG,SAAS,GAAG,UAAU,CAAC;AAE/D,MAAM,WAAW,cAAc;IAC7B,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,6EAA6E;IAC7E,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAmDD,MAAM,WAAW,cAAc;IAC7B,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,mFAAmF;AACnF,wBAAgB,aAAa,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,GAAE,MAAuB,EAAE,GAAG,GAAE,MAAmB,GAAG,cAAc,CAavH;AAED,MAAM,MAAM,WAAW,GACnB;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,GAC/B;IAAE,EAAE,EAAE,KAAK,CAAC;IAAC,MAAM,EAAE,WAAW,GAAG,SAAS,GAAG,iBAAiB,CAAA;CAAE,CAAC;AAEvE,4FAA4F;AAC5F,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,EAAE,GAAG,GAAE,MAAmB,GAAG,WAAW,CAe1H;AAED,MAAM,MAAM,WAAW,GACnB;IAAE,MAAM,EAAE,SAAS,CAAA;CAAE,GACrB;IAAE,MAAM,EAAE,SAAS,CAAC;IAAC,SAAS,EAAE,cAAc,CAAA;CAAE,GAChD;IAAE,MAAM,EAAE,UAAU,CAAA;CAAE,GACtB;IAAE,MAAM,EAAE,WAAW,CAAA;CAAE,CAAC;AAE5B,oEAAoE;AACpE,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,GAAE,MAAmB,GAAG,WAAW,CAOhI;AAED,MAAM,MAAM,aAAa,GACrB;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,SAAS,EAAE,cAAc,CAAA;CAAE,GACvC;IAAE,EAAE,EAAE,KAAK,CAAC;IAAC,MAAM,EAAE,WAAW,GAAG,aAAa,GAAG,kBAAkB,CAAA;CAAE,CAAC;AAE5E,oFAAoF;AACpF,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,GAAE,MAAmB,GAAG,aAAa,CAUhI"}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pairing-login store: binds a browser session to a Feishu user without a web
|
|
3
|
+
* OAuth redirect (which is unfriendly to ip:port self-host — see
|
|
4
|
+
* docs/platform-design.md). Device-code style:
|
|
5
|
+
*
|
|
6
|
+
* 1. Browser (unauthenticated) calls start → gets a short human `code` to show
|
|
7
|
+
* the user, plus a high-entropy `browserToken` it keeps privately.
|
|
8
|
+
* 2. User sends the `code` to the bot in Feishu. The daemon (which already
|
|
9
|
+
* knows the sender's open_id) claims the pairing for that identity.
|
|
10
|
+
* 3. Browser polls/consumes with its `browserToken`; on a claimed pairing it
|
|
11
|
+
* learns the Feishu identity and the web endpoint issues a session.
|
|
12
|
+
*
|
|
13
|
+
* Identity is established INSIDE Feishu, so only a short-lived code crosses to
|
|
14
|
+
* the web — independent of domain/IP/port. Team-membership gating is the
|
|
15
|
+
* caller's job (via team-store); this store only pairs browser ↔ identity.
|
|
16
|
+
*
|
|
17
|
+
* Security: code is high-entropy + short TTL + single-use; browserToken gates
|
|
18
|
+
* status/consume so a guessed code alone can't hijack a browser session.
|
|
19
|
+
* (Brute-forcing codes is further bounded by endpoint rate limiting at the wiring
|
|
20
|
+
* layer.) Codes/tokens are never logged.
|
|
21
|
+
*
|
|
22
|
+
* Storage: `{dataDir}/pairings.json` (shared across the web + daemon processes),
|
|
23
|
+
* atomic writes.
|
|
24
|
+
*/
|
|
25
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync, renameSync } from 'node:fs';
|
|
26
|
+
import { join } from 'node:path';
|
|
27
|
+
import { randomBytes, randomUUID } from 'node:crypto';
|
|
28
|
+
const DEFAULT_TTL_MS = 5 * 60 * 1000; // 5 minutes
|
|
29
|
+
/** Unambiguous alphabet (no 0/O/1/I) for the human-entered code. */
|
|
30
|
+
const CODE_ALPHABET = 'ABCDEFGHJKLMNPQRSTUVWXYZ23456789';
|
|
31
|
+
const CODE_LEN = 8;
|
|
32
|
+
function filePath(dataDir) {
|
|
33
|
+
return join(dataDir, 'pairings.json');
|
|
34
|
+
}
|
|
35
|
+
function readFile(dataDir) {
|
|
36
|
+
const fp = filePath(dataDir);
|
|
37
|
+
if (!existsSync(fp))
|
|
38
|
+
return {};
|
|
39
|
+
try {
|
|
40
|
+
const parsed = JSON.parse(readFileSync(fp, 'utf-8'));
|
|
41
|
+
if (parsed && typeof parsed === 'object' && !Array.isArray(parsed))
|
|
42
|
+
return parsed;
|
|
43
|
+
}
|
|
44
|
+
catch { /* corrupt — fall through */ }
|
|
45
|
+
return {};
|
|
46
|
+
}
|
|
47
|
+
function writeFileAtomic(dataDir, data) {
|
|
48
|
+
if (!existsSync(dataDir))
|
|
49
|
+
mkdirSync(dataDir, { recursive: true });
|
|
50
|
+
const fp = filePath(dataDir);
|
|
51
|
+
const tmp = `${fp}.${process.pid}.${randomUUID()}.tmp`;
|
|
52
|
+
writeFileSync(tmp, JSON.stringify(data, null, 2) + '\n', 'utf-8');
|
|
53
|
+
renameSync(tmp, fp);
|
|
54
|
+
}
|
|
55
|
+
/** Drop expired entries; returns the live map. */
|
|
56
|
+
function prune(data, now) {
|
|
57
|
+
for (const [id, e] of Object.entries(data)) {
|
|
58
|
+
if (e.expiresAt <= now)
|
|
59
|
+
delete data[id];
|
|
60
|
+
}
|
|
61
|
+
return data;
|
|
62
|
+
}
|
|
63
|
+
function genCode() {
|
|
64
|
+
const bytes = randomBytes(CODE_LEN);
|
|
65
|
+
let out = '';
|
|
66
|
+
for (let i = 0; i < CODE_LEN; i++)
|
|
67
|
+
out += CODE_ALPHABET[bytes[i] % CODE_ALPHABET.length];
|
|
68
|
+
return out;
|
|
69
|
+
}
|
|
70
|
+
/** Begin a pairing. Returns the code to show the user + a private browserToken. */
|
|
71
|
+
export function createPairing(dataDir, ttlMs = DEFAULT_TTL_MS, now = Date.now()) {
|
|
72
|
+
const data = prune(readFile(dataDir), now);
|
|
73
|
+
const entry = {
|
|
74
|
+
pairingId: randomUUID(),
|
|
75
|
+
code: genCode(),
|
|
76
|
+
browserToken: randomBytes(24).toString('base64url'),
|
|
77
|
+
status: 'pending',
|
|
78
|
+
createdAt: now,
|
|
79
|
+
expiresAt: now + ttlMs,
|
|
80
|
+
};
|
|
81
|
+
data[entry.pairingId] = entry;
|
|
82
|
+
writeFileAtomic(dataDir, data);
|
|
83
|
+
return { pairingId: entry.pairingId, code: entry.code, browserToken: entry.browserToken, expiresAt: entry.expiresAt };
|
|
84
|
+
}
|
|
85
|
+
/** Claim a pending pairing for a Feishu identity (called by the daemon on the bot side). */
|
|
86
|
+
export function claimPairing(dataDir, code, claimer, now = Date.now()) {
|
|
87
|
+
const data = prune(readFile(dataDir), now);
|
|
88
|
+
const entry = Object.values(data).find(e => e.code === code.trim().toUpperCase());
|
|
89
|
+
if (!entry)
|
|
90
|
+
return { ok: false, reason: 'not_found' };
|
|
91
|
+
if (entry.expiresAt <= now)
|
|
92
|
+
return { ok: false, reason: 'expired' };
|
|
93
|
+
if (entry.status !== 'pending')
|
|
94
|
+
return { ok: false, reason: 'already_claimed' };
|
|
95
|
+
entry.status = 'claimed';
|
|
96
|
+
entry.claimedBy = {
|
|
97
|
+
openId: claimer.openId,
|
|
98
|
+
...(claimer.unionId ? { unionId: claimer.unionId } : {}),
|
|
99
|
+
...(claimer.name ? { name: claimer.name } : {}),
|
|
100
|
+
...(claimer.larkAppId ? { larkAppId: claimer.larkAppId } : {}),
|
|
101
|
+
};
|
|
102
|
+
writeFileAtomic(dataDir, data);
|
|
103
|
+
return { ok: true, pairingId: entry.pairingId };
|
|
104
|
+
}
|
|
105
|
+
/** Browser-side status poll; requires the matching browserToken. */
|
|
106
|
+
export function getPairingStatus(dataDir, pairingId, browserToken, now = Date.now()) {
|
|
107
|
+
const data = prune(readFile(dataDir), now);
|
|
108
|
+
const entry = data[pairingId];
|
|
109
|
+
if (!entry || entry.browserToken !== browserToken)
|
|
110
|
+
return { status: 'not_found' };
|
|
111
|
+
if (entry.status === 'claimed' && entry.claimedBy)
|
|
112
|
+
return { status: 'claimed', claimedBy: entry.claimedBy };
|
|
113
|
+
if (entry.status === 'consumed')
|
|
114
|
+
return { status: 'consumed' };
|
|
115
|
+
return { status: 'pending' };
|
|
116
|
+
}
|
|
117
|
+
/** Single-use: turn a claimed pairing into a session. Requires the browserToken. */
|
|
118
|
+
export function consumePairing(dataDir, pairingId, browserToken, now = Date.now()) {
|
|
119
|
+
const data = prune(readFile(dataDir), now);
|
|
120
|
+
const entry = data[pairingId];
|
|
121
|
+
if (!entry || entry.browserToken !== browserToken)
|
|
122
|
+
return { ok: false, reason: 'not_found' };
|
|
123
|
+
if (entry.status === 'consumed')
|
|
124
|
+
return { ok: false, reason: 'already_consumed' };
|
|
125
|
+
if (entry.status !== 'claimed' || !entry.claimedBy)
|
|
126
|
+
return { ok: false, reason: 'not_claimed' };
|
|
127
|
+
entry.status = 'consumed';
|
|
128
|
+
const claimedBy = entry.claimedBy;
|
|
129
|
+
writeFileAtomic(dataDir, data);
|
|
130
|
+
return { ok: true, claimedBy };
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=pairing-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pairing-store.js","sourceRoot":"","sources":["../../src/services/pairing-store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared helper for collecting the operator's relayable sessions used by the
|
|
3
|
+
* /relay picker. Same selection criteria used both at first render
|
|
4
|
+
* (command-handler) and on each card-update click (card-handler re-render),
|
|
5
|
+
* so factor it out to keep both paths in sync.
|
|
6
|
+
*
|
|
7
|
+
* Selection rules:
|
|
8
|
+
* • same bot (this larkAppId — only this daemon's sessions are visible)
|
|
9
|
+
* • NOT in the current chat (can't relay into the chat it already lives in)
|
|
10
|
+
* • operator is the session owner (owner-only access)
|
|
11
|
+
* • not an adopt session (those wrap a user-attached tmux pane, refused
|
|
12
|
+
* by transferSession anyway)
|
|
13
|
+
*
|
|
14
|
+
* Resolves friendly chat names and modes via getChatNameAndMode in parallel
|
|
15
|
+
* (1 API call per unique source chatId). Failure modes are tolerant:
|
|
16
|
+
* unresolved chats fall back to the raw chatId for chatLabel and the
|
|
17
|
+
* session's own chatType for mode.
|
|
18
|
+
*/
|
|
19
|
+
import type { DaemonSession } from '../core/types.js';
|
|
20
|
+
import type { RelayPickerEntry } from '../im/lark/card-builder.js';
|
|
21
|
+
export declare function collectRelayPickerEntries(activeSessions: Map<string, DaemonSession>, myAppId: string, currentChatId: string, operatorOpenId: string): Promise<RelayPickerEntry[]>;
|
|
22
|
+
//# sourceMappingURL=relay-picker.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"relay-picker.d.ts","sourceRoot":"","sources":["../../src/services/relay-picker.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AACH,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACtD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,4BAA4B,CAAC;AAInE,wBAAsB,yBAAyB,CAC7C,cAAc,EAAE,GAAG,CAAC,MAAM,EAAE,aAAa,CAAC,EAC1C,OAAO,EAAE,MAAM,EACf,aAAa,EAAE,MAAM,EACrB,cAAc,EAAE,MAAM,GACrB,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAwD7B"}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { getChatNameAndMode } from '../im/lark/client.js';
|
|
2
|
+
import { isRelayableRealSession } from '../core/worker-pool.js';
|
|
3
|
+
export async function collectRelayPickerEntries(activeSessions, myAppId, currentChatId, operatorOpenId) {
|
|
4
|
+
const candidates = [];
|
|
5
|
+
for (const c of activeSessions.values()) {
|
|
6
|
+
if (c.larkAppId !== myAppId)
|
|
7
|
+
continue;
|
|
8
|
+
if (c.chatId === currentChatId)
|
|
9
|
+
continue;
|
|
10
|
+
if (c.session.ownerOpenId !== operatorOpenId)
|
|
11
|
+
continue;
|
|
12
|
+
if (c.session.adoptedFrom)
|
|
13
|
+
continue;
|
|
14
|
+
// Daemon-command scratches (worker:null + no persisted CLI markers)
|
|
15
|
+
// are placeholder records for /help / unfinished /relay etc. — they
|
|
16
|
+
// have no real conversation to bring along. Don't surface them in
|
|
17
|
+
// anyone's picker.
|
|
18
|
+
if (!isRelayableRealSession(c))
|
|
19
|
+
continue;
|
|
20
|
+
candidates.push(c);
|
|
21
|
+
}
|
|
22
|
+
// Skip the API call entirely for p2p chats. session.chatType is recorded
|
|
23
|
+
// at session creation from the Lark event payload and is authoritative —
|
|
24
|
+
// it doesn't drift. The earlier design used `info?.mode ?? fallbackMode`,
|
|
25
|
+
// but `getChatNameAndMode` swallows API errors and returns the SAFE
|
|
26
|
+
// DEFAULT `{ name: null, mode: 'group' }` — which then mis-classified
|
|
27
|
+
// every p2p session as 普通群 whenever the chat.get call failed
|
|
28
|
+
// (permissions / network / etc.). 王皓 caught this. Authoritative path:
|
|
29
|
+
// p2p → session.chatType; non-p2p → Lark API (for group/topic split).
|
|
30
|
+
const groupChatIds = [...new Set(candidates.filter(c => c.chatType !== 'p2p').map(c => c.chatId))];
|
|
31
|
+
const resolved = await Promise.all(groupChatIds.map(async (cid) => [cid, await getChatNameAndMode(myAppId, cid)]));
|
|
32
|
+
const chatInfo = new Map();
|
|
33
|
+
for (const [cid, info] of resolved)
|
|
34
|
+
chatInfo.set(cid, info);
|
|
35
|
+
return candidates.map(c => {
|
|
36
|
+
if (c.chatType === 'p2p') {
|
|
37
|
+
return {
|
|
38
|
+
sessionId: c.session.sessionId,
|
|
39
|
+
// chatLabel is unused for p2p in the rendered output (location
|
|
40
|
+
// field always renders the locale-aware "单聊" literal), but we
|
|
41
|
+
// still set it to chatId for completeness / downstream debug.
|
|
42
|
+
chatLabel: c.chatId,
|
|
43
|
+
title: c.session.title || c.currentTurnTitle || '(no title)',
|
|
44
|
+
workingDir: c.session.workingDir,
|
|
45
|
+
cliId: c.session.cliId,
|
|
46
|
+
lastMessageAt: c.lastMessageAt,
|
|
47
|
+
chatMode: 'p2p',
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
const info = chatInfo.get(c.chatId);
|
|
51
|
+
return {
|
|
52
|
+
sessionId: c.session.sessionId,
|
|
53
|
+
chatLabel: info?.name ?? c.chatId,
|
|
54
|
+
title: c.session.title || c.currentTurnTitle || '(no title)',
|
|
55
|
+
workingDir: c.session.workingDir,
|
|
56
|
+
cliId: c.session.cliId,
|
|
57
|
+
lastMessageAt: c.lastMessageAt,
|
|
58
|
+
chatMode: info?.mode ?? 'group',
|
|
59
|
+
};
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=relay-picker.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"relay-picker.js","sourceRoot":"","sources":["../../src/services/relay-picker.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
export interface TeamRosterBot {
|
|
2
|
+
larkAppId: string;
|
|
3
|
+
name: string;
|
|
4
|
+
cliId: string;
|
|
5
|
+
capability: string | null;
|
|
6
|
+
hasTeamRole: boolean;
|
|
7
|
+
/** Owner for grouping by person; null if unassigned. unionId is the key, name for display. */
|
|
8
|
+
owner: {
|
|
9
|
+
unionId?: string;
|
|
10
|
+
openId?: string;
|
|
11
|
+
name?: string;
|
|
12
|
+
} | null;
|
|
13
|
+
}
|
|
14
|
+
export interface TeamRoster {
|
|
15
|
+
team: {
|
|
16
|
+
id: string;
|
|
17
|
+
name: string;
|
|
18
|
+
memberCount: number;
|
|
19
|
+
};
|
|
20
|
+
bots: TeamRosterBot[];
|
|
21
|
+
}
|
|
22
|
+
/** A currently-running bot from the live daemon registry (authoritative for
|
|
23
|
+
* "what's running" — unlike bots-info.json which is a racy, probe-lagged file). */
|
|
24
|
+
export interface LiveBot {
|
|
25
|
+
larkAppId: string;
|
|
26
|
+
botName?: string | null;
|
|
27
|
+
cliId?: string;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* @param configOrder optional list of larkAppIds in bots.json (config) order;
|
|
31
|
+
* when given, the roster is sorted to match it (and the personal dashboard).
|
|
32
|
+
* @param liveBots optional live daemon-registry bots. When given, the roster's
|
|
33
|
+
* bot set is THESE (authoritative — fixes an empty/stale bots-info.json
|
|
34
|
+
* showing no bots even though daemons are running), enriched with cliId from
|
|
35
|
+
* bots-info.json by larkAppId. When omitted, falls back to bots-info.json.
|
|
36
|
+
*/
|
|
37
|
+
export declare function buildTeamRoster(dataDir: string, teamId?: string, configOrder?: string[], liveBots?: LiveBot[]): TeamRoster;
|
|
38
|
+
//# sourceMappingURL=team-roster.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"team-roster.d.ts","sourceRoot":"","sources":["../../src/services/team-roster.ts"],"names":[],"mappings":"AAgBA,MAAM,WAAW,aAAa;IAC5B,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,WAAW,EAAE,OAAO,CAAC;IACrB,8FAA8F;IAC9F,KAAK,EAAE;QAAE,OAAO,CAAC,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;CACpE;AAED,MAAM,WAAW,UAAU;IACzB,IAAI,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAA;KAAE,CAAC;IACxD,IAAI,EAAE,aAAa,EAAE,CAAC;CACvB;AAkBD;oFACoF;AACpF,MAAM,WAAW,OAAO;IAAG,SAAS,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE;AAEvF;;;;;;;GAOG;AACH,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,GAAE,MAAwB,EAAE,WAAW,CAAC,EAAE,MAAM,EAAE,EAAE,QAAQ,CAAC,EAAE,OAAO,EAAE,GAAG,UAAU,CAyC3I"}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Team-level collaboration roster for the platform UI: every bot the deployment
|
|
3
|
+
* runs (from bots-info.json) enriched with its team capability label and whether
|
|
4
|
+
* it has a team-level role, plus the team's member count.
|
|
5
|
+
*
|
|
6
|
+
* This is the TEAM view (who's on the team), distinct from the per-chat roster
|
|
7
|
+
* in listChatBotMembers (who's in a given group + reliably @-mentionable).
|
|
8
|
+
* Pure read from `{dataDir}` files — no Lark API, no config coupling — so it is
|
|
9
|
+
* trivially testable and cheap for the UI to poll.
|
|
10
|
+
*/
|
|
11
|
+
import { readFileSync, existsSync } from 'node:fs';
|
|
12
|
+
import { join } from 'node:path';
|
|
13
|
+
import { getBotCapability } from './bot-profile-store.js';
|
|
14
|
+
import { getBotOwner } from './bot-owner-store.js';
|
|
15
|
+
import { getTeam, getDefaultTeam, DEFAULT_TEAM_ID } from './team-store.js';
|
|
16
|
+
function readBotsInfo(dataDir) {
|
|
17
|
+
const fp = join(dataDir, 'bots-info.json');
|
|
18
|
+
if (!existsSync(fp))
|
|
19
|
+
return [];
|
|
20
|
+
try {
|
|
21
|
+
const parsed = JSON.parse(readFileSync(fp, 'utf-8'));
|
|
22
|
+
if (Array.isArray(parsed))
|
|
23
|
+
return parsed;
|
|
24
|
+
}
|
|
25
|
+
catch { /* corrupt — empty roster */ }
|
|
26
|
+
return [];
|
|
27
|
+
}
|
|
28
|
+
function hasTeamRoleFile(dataDir, larkAppId) {
|
|
29
|
+
return existsSync(join(dataDir, 'team-roles', `${larkAppId}.md`));
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* @param configOrder optional list of larkAppIds in bots.json (config) order;
|
|
33
|
+
* when given, the roster is sorted to match it (and the personal dashboard).
|
|
34
|
+
* @param liveBots optional live daemon-registry bots. When given, the roster's
|
|
35
|
+
* bot set is THESE (authoritative — fixes an empty/stale bots-info.json
|
|
36
|
+
* showing no bots even though daemons are running), enriched with cliId from
|
|
37
|
+
* bots-info.json by larkAppId. When omitted, falls back to bots-info.json.
|
|
38
|
+
*/
|
|
39
|
+
export function buildTeamRoster(dataDir, teamId = DEFAULT_TEAM_ID, configOrder, liveBots) {
|
|
40
|
+
const team = getTeam(dataDir, teamId) ?? getDefaultTeam(dataDir);
|
|
41
|
+
const info = readBotsInfo(dataDir);
|
|
42
|
+
let entries;
|
|
43
|
+
if (liveBots !== undefined) {
|
|
44
|
+
// Live registry is the source of truth WHEN PROVIDED — including an empty
|
|
45
|
+
// array (no daemons running ⇒ empty roster). Never fall back to a stale
|
|
46
|
+
// bots-info.json here, or removed/offline bots would linger. Enrich
|
|
47
|
+
// cliId/openId/name from bots-info.json (which carries cliId — the registry
|
|
48
|
+
// doesn't) by larkAppId.
|
|
49
|
+
const byId = new Map(info.map(e => [e.larkAppId, e]));
|
|
50
|
+
entries = liveBots.map(lb => {
|
|
51
|
+
const ex = byId.get(lb.larkAppId);
|
|
52
|
+
return {
|
|
53
|
+
larkAppId: lb.larkAppId,
|
|
54
|
+
botOpenId: ex?.botOpenId ?? null,
|
|
55
|
+
botName: lb.botName ?? ex?.botName ?? null,
|
|
56
|
+
cliId: lb.cliId ?? ex?.cliId ?? '',
|
|
57
|
+
};
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
else {
|
|
61
|
+
entries = info;
|
|
62
|
+
}
|
|
63
|
+
if (configOrder && configOrder.length) {
|
|
64
|
+
const rank = new Map(configOrder.map((id, i) => [id, i]));
|
|
65
|
+
const at = (id) => rank.has(id) ? rank.get(id) : Number.MAX_SAFE_INTEGER;
|
|
66
|
+
// stable sort by config index; unknown bots fall to the end keeping their order
|
|
67
|
+
entries = entries.map((b, i) => ({ b, i })).sort((x, y) => (at(x.b.larkAppId) - at(y.b.larkAppId)) || (x.i - y.i)).map(x => x.b);
|
|
68
|
+
}
|
|
69
|
+
const bots = entries.map((b) => {
|
|
70
|
+
const o = getBotOwner(dataDir, b.larkAppId);
|
|
71
|
+
return {
|
|
72
|
+
larkAppId: b.larkAppId,
|
|
73
|
+
name: b.botName ?? b.cliId,
|
|
74
|
+
cliId: b.cliId,
|
|
75
|
+
capability: getBotCapability(dataDir, b.larkAppId),
|
|
76
|
+
hasTeamRole: hasTeamRoleFile(dataDir, b.larkAppId),
|
|
77
|
+
owner: o ? { unionId: o.unionId, openId: o.openId, name: o.name } : null,
|
|
78
|
+
};
|
|
79
|
+
});
|
|
80
|
+
return { team: { id: team.id, name: team.name, memberCount: team.members.length }, bots };
|
|
81
|
+
}
|
|
82
|
+
//# sourceMappingURL=team-roster.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"team-roster.js","sourceRoot":"","sources":["../../src/services/team-roster.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
export declare const DEFAULT_TEAM_ID = "default";
|
|
2
|
+
export interface TeamMember {
|
|
3
|
+
unionId?: string;
|
|
4
|
+
openId?: string;
|
|
5
|
+
email?: string;
|
|
6
|
+
name?: string;
|
|
7
|
+
addedAt: number;
|
|
8
|
+
addedBy?: string;
|
|
9
|
+
}
|
|
10
|
+
export interface Team {
|
|
11
|
+
id: string;
|
|
12
|
+
name: string;
|
|
13
|
+
members: TeamMember[];
|
|
14
|
+
createdAt: number;
|
|
15
|
+
updatedAt: number;
|
|
16
|
+
}
|
|
17
|
+
/** Identifiers a membership check can match on. Match succeeds on ANY non-empty hit. */
|
|
18
|
+
export interface MemberIdentity {
|
|
19
|
+
unionId?: string;
|
|
20
|
+
openId?: string;
|
|
21
|
+
email?: string;
|
|
22
|
+
}
|
|
23
|
+
/** All teams (empty array if none materialized yet). */
|
|
24
|
+
export declare function listTeams(dataDir: string): Team[];
|
|
25
|
+
/** A team by id, or null. */
|
|
26
|
+
export declare function getTeam(dataDir: string, teamId: string): Team | null;
|
|
27
|
+
/** All teams the given identity belongs to (matches any identifier). */
|
|
28
|
+
export declare function listTeamsForMember(dataDir: string, id: MemberIdentity): Team[];
|
|
29
|
+
/**
|
|
30
|
+
* The default (implicit single-deployment) team. Read-only: returns the first
|
|
31
|
+
* persisted team, else a synthesized empty default that is NOT written. Use
|
|
32
|
+
* `ensureDefaultTeam` when you intend to persist.
|
|
33
|
+
*/
|
|
34
|
+
export declare function getDefaultTeam(dataDir: string): Team;
|
|
35
|
+
/** Materialize (persist) the default team if no team exists yet; returns it. */
|
|
36
|
+
export declare function ensureDefaultTeam(dataDir: string, now?: number): Team;
|
|
37
|
+
/** Delete a team entirely (members + the team entry). Returns true if removed.
|
|
38
|
+
* Bots/connectors live at the deployment level (not per-team), so they are
|
|
39
|
+
* unaffected — only the membership/trust boundary is dropped. */
|
|
40
|
+
export declare function deleteTeam(dataDir: string, teamId: string): boolean;
|
|
41
|
+
/** Create a new explicit team. */
|
|
42
|
+
export declare function createTeam(dataDir: string, name: string, now?: number): Team;
|
|
43
|
+
/**
|
|
44
|
+
* Add (or merge) a member into a team. Dedupes by any matching identifier and
|
|
45
|
+
* fills in newly-known fields. Returns the updated team, or null if no such team.
|
|
46
|
+
*/
|
|
47
|
+
export declare function addMember(dataDir: string, teamId: string, member: Omit<TeamMember, 'addedAt'> & {
|
|
48
|
+
addedAt?: number;
|
|
49
|
+
}, now?: number): Team | null;
|
|
50
|
+
/** Remove a member matching any identifier. Returns true if removed. */
|
|
51
|
+
export declare function removeMember(dataDir: string, teamId: string, id: MemberIdentity, now?: number): boolean;
|
|
52
|
+
/** Whether the given identity is a member of the team (matches any identifier). */
|
|
53
|
+
export declare function isMember(dataDir: string, teamId: string, id: MemberIdentity): boolean;
|
|
54
|
+
//# sourceMappingURL=team-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"team-store.d.ts","sourceRoot":"","sources":["../../src/services/team-store.ts"],"names":[],"mappings":"AAuBA,eAAO,MAAM,eAAe,YAAY,CAAC;AAEzC,MAAM,WAAW,UAAU;IACzB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,IAAI;IACnB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,UAAU,EAAE,CAAC;IACtB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAOD,wFAAwF;AACxF,MAAM,WAAW,cAAc;IAC7B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAwBD,wDAAwD;AACxD,wBAAgB,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,EAAE,CAEjD;AAED,6BAA6B;AAC7B,wBAAgB,OAAO,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI,CAEpE;AAED,wEAAwE;AACxE,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,EAAE,cAAc,GAAG,IAAI,EAAE,CAG9E;AAED;;;;GAIG;AACH,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAMpD;AAED,gFAAgF;AAChF,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,GAAE,MAAmB,GAAG,IAAI,CAOjF;AAED;;kEAEkE;AAClE,wBAAgB,UAAU,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAOnE;AAED,kCAAkC;AAClC,wBAAgB,UAAU,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,GAAE,MAAmB,GAAG,IAAI,CAMxF;AAQD;;;GAGG;AACH,wBAAgB,SAAS,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,CAAC,UAAU,EAAE,SAAS,CAAC,GAAG;IAAE,OAAO,CAAC,EAAE,MAAM,CAAA;CAAE,EAAE,GAAG,GAAE,MAAmB,GAAG,IAAI,GAAG,IAAI,CAgB5J;AAED,wEAAwE;AACxE,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,EAAE,cAAc,EAAE,GAAG,GAAE,MAAmB,GAAG,OAAO,CAUnH;AAED,mFAAmF;AACnF,wBAAgB,QAAQ,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,EAAE,cAAc,GAAG,OAAO,CAKrF"}
|