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.
Files changed (192) hide show
  1. package/README.en.md +1 -2
  2. package/README.md +1 -2
  3. package/dist/cli/bots-list-output.d.ts +8 -0
  4. package/dist/cli/bots-list-output.d.ts.map +1 -1
  5. package/dist/cli/bots-list-output.js +9 -0
  6. package/dist/cli/bots-list-output.js.map +1 -1
  7. package/dist/cli.d.ts.map +1 -1
  8. package/dist/cli.js +6 -0
  9. package/dist/cli.js.map +1 -1
  10. package/dist/core/command-handler.d.ts.map +1 -1
  11. package/dist/core/command-handler.js +601 -50
  12. package/dist/core/command-handler.js.map +1 -1
  13. package/dist/core/dashboard-ipc-server.d.ts +2 -0
  14. package/dist/core/dashboard-ipc-server.d.ts.map +1 -1
  15. package/dist/core/dashboard-ipc-server.js +170 -2
  16. package/dist/core/dashboard-ipc-server.js.map +1 -1
  17. package/dist/core/role-resolver.d.ts +17 -1
  18. package/dist/core/role-resolver.d.ts.map +1 -1
  19. package/dist/core/role-resolver.js +64 -10
  20. package/dist/core/role-resolver.js.map +1 -1
  21. package/dist/core/session-manager.d.ts +1 -1
  22. package/dist/core/session-manager.d.ts.map +1 -1
  23. package/dist/core/session-manager.js +24 -13
  24. package/dist/core/session-manager.js.map +1 -1
  25. package/dist/core/trigger-session.d.ts +9 -0
  26. package/dist/core/trigger-session.d.ts.map +1 -0
  27. package/dist/core/trigger-session.js +158 -0
  28. package/dist/core/trigger-session.js.map +1 -0
  29. package/dist/core/worker-pool.d.ts +105 -0
  30. package/dist/core/worker-pool.d.ts.map +1 -1
  31. package/dist/core/worker-pool.js +284 -4
  32. package/dist/core/worker-pool.js.map +1 -1
  33. package/dist/daemon.d.ts.map +1 -1
  34. package/dist/daemon.js +80 -45
  35. package/dist/daemon.js.map +1 -1
  36. package/dist/dashboard/connector-api.d.ts +3 -0
  37. package/dist/dashboard/connector-api.d.ts.map +1 -0
  38. package/dist/dashboard/connector-api.js +351 -0
  39. package/dist/dashboard/connector-api.js.map +1 -0
  40. package/dist/dashboard/federated-group-core.d.ts +54 -0
  41. package/dist/dashboard/federated-group-core.d.ts.map +1 -0
  42. package/dist/dashboard/federated-group-core.js +165 -0
  43. package/dist/dashboard/federated-group-core.js.map +1 -0
  44. package/dist/dashboard/federation-api.d.ts +42 -0
  45. package/dist/dashboard/federation-api.d.ts.map +1 -0
  46. package/dist/dashboard/federation-api.js +408 -0
  47. package/dist/dashboard/federation-api.js.map +1 -0
  48. package/dist/dashboard/federation-spoke-api.d.ts +76 -0
  49. package/dist/dashboard/federation-spoke-api.d.ts.map +1 -0
  50. package/dist/dashboard/federation-spoke-api.js +618 -0
  51. package/dist/dashboard/federation-spoke-api.js.map +1 -0
  52. package/dist/dashboard/team-group.d.ts +18 -0
  53. package/dist/dashboard/team-group.d.ts.map +1 -0
  54. package/dist/dashboard/team-group.js +7 -0
  55. package/dist/dashboard/team-group.js.map +1 -0
  56. package/dist/dashboard/trigger-api.d.ts +13 -0
  57. package/dist/dashboard/trigger-api.d.ts.map +1 -0
  58. package/dist/dashboard/trigger-api.js +77 -0
  59. package/dist/dashboard/trigger-api.js.map +1 -0
  60. package/dist/dashboard/web/app.js +8 -0
  61. package/dist/dashboard/web/app.js.map +1 -1
  62. package/dist/dashboard/web/connectors.d.ts +2 -0
  63. package/dist/dashboard/web/connectors.d.ts.map +1 -0
  64. package/dist/dashboard/web/connectors.js +187 -0
  65. package/dist/dashboard/web/connectors.js.map +1 -0
  66. package/dist/dashboard/web/team-federation.d.ts +3 -0
  67. package/dist/dashboard/web/team-federation.d.ts.map +1 -0
  68. package/dist/dashboard/web/team-federation.js +487 -0
  69. package/dist/dashboard/web/team-federation.js.map +1 -0
  70. package/dist/dashboard/webhook-routes.d.ts +19 -0
  71. package/dist/dashboard/webhook-routes.d.ts.map +1 -0
  72. package/dist/dashboard/webhook-routes.js +321 -0
  73. package/dist/dashboard/webhook-routes.js.map +1 -0
  74. package/dist/dashboard-web/app.js +509 -386
  75. package/dist/dashboard-web/index.html +2 -0
  76. package/dist/dashboard.js +152 -1
  77. package/dist/dashboard.js.map +1 -1
  78. package/dist/i18n/en.d.ts.map +1 -1
  79. package/dist/i18n/en.js +85 -8
  80. package/dist/i18n/en.js.map +1 -1
  81. package/dist/i18n/zh.d.ts.map +1 -1
  82. package/dist/i18n/zh.js +85 -8
  83. package/dist/i18n/zh.js.map +1 -1
  84. package/dist/im/lark/card-builder.d.ts +82 -0
  85. package/dist/im/lark/card-builder.d.ts.map +1 -1
  86. package/dist/im/lark/card-builder.js +315 -0
  87. package/dist/im/lark/card-builder.js.map +1 -1
  88. package/dist/im/lark/card-handler.d.ts.map +1 -1
  89. package/dist/im/lark/card-handler.js +195 -1
  90. package/dist/im/lark/card-handler.js.map +1 -1
  91. package/dist/im/lark/client.d.ts +45 -0
  92. package/dist/im/lark/client.d.ts.map +1 -1
  93. package/dist/im/lark/client.js +169 -18
  94. package/dist/im/lark/client.js.map +1 -1
  95. package/dist/im/lark/message-parser.d.ts.map +1 -1
  96. package/dist/im/lark/message-parser.js +1 -0
  97. package/dist/im/lark/message-parser.js.map +1 -1
  98. package/dist/services/bot-owner-store.d.ts +28 -0
  99. package/dist/services/bot-owner-store.d.ts.map +1 -0
  100. package/dist/services/bot-owner-store.js +82 -0
  101. package/dist/services/bot-owner-store.js.map +1 -0
  102. package/dist/services/bot-profile-store.d.ts +16 -0
  103. package/dist/services/bot-profile-store.d.ts.map +1 -0
  104. package/dist/services/bot-profile-store.js +98 -0
  105. package/dist/services/bot-profile-store.js.map +1 -0
  106. package/dist/services/connector-store.d.ts +58 -0
  107. package/dist/services/connector-store.d.ts.map +1 -0
  108. package/dist/services/connector-store.js +79 -0
  109. package/dist/services/connector-store.js.map +1 -0
  110. package/dist/services/deployment-identity.d.ts +22 -0
  111. package/dist/services/deployment-identity.d.ts.map +1 -0
  112. package/dist/services/deployment-identity.js +67 -0
  113. package/dist/services/deployment-identity.js.map +1 -0
  114. package/dist/services/federation-membership-store.d.ts +23 -0
  115. package/dist/services/federation-membership-store.d.ts.map +1 -0
  116. package/dist/services/federation-membership-store.js +66 -0
  117. package/dist/services/federation-membership-store.js.map +1 -0
  118. package/dist/services/federation-roster.d.ts +54 -0
  119. package/dist/services/federation-roster.d.ts.map +1 -0
  120. package/dist/services/federation-roster.js +51 -0
  121. package/dist/services/federation-roster.js.map +1 -0
  122. package/dist/services/federation-store.d.ts +76 -0
  123. package/dist/services/federation-store.d.ts.map +1 -0
  124. package/dist/services/federation-store.js +133 -0
  125. package/dist/services/federation-store.js.map +1 -0
  126. package/dist/services/group-creator.d.ts +6 -0
  127. package/dist/services/group-creator.d.ts.map +1 -1
  128. package/dist/services/group-creator.js +10 -1
  129. package/dist/services/group-creator.js.map +1 -1
  130. package/dist/services/groups-store.d.ts +11 -0
  131. package/dist/services/groups-store.d.ts.map +1 -1
  132. package/dist/services/groups-store.js +42 -0
  133. package/dist/services/groups-store.js.map +1 -1
  134. package/dist/services/invite-store.d.ts +28 -0
  135. package/dist/services/invite-store.d.ts.map +1 -0
  136. package/dist/services/invite-store.js +85 -0
  137. package/dist/services/invite-store.js.map +1 -0
  138. package/dist/services/pairing-store.d.ts +47 -0
  139. package/dist/services/pairing-store.d.ts.map +1 -0
  140. package/dist/services/pairing-store.js +132 -0
  141. package/dist/services/pairing-store.js.map +1 -0
  142. package/dist/services/relay-picker.d.ts +22 -0
  143. package/dist/services/relay-picker.d.ts.map +1 -0
  144. package/dist/services/relay-picker.js +62 -0
  145. package/dist/services/relay-picker.js.map +1 -0
  146. package/dist/services/team-roster.d.ts +38 -0
  147. package/dist/services/team-roster.d.ts.map +1 -0
  148. package/dist/services/team-roster.js +82 -0
  149. package/dist/services/team-roster.js.map +1 -0
  150. package/dist/services/team-store.d.ts +54 -0
  151. package/dist/services/team-store.d.ts.map +1 -0
  152. package/dist/services/team-store.js +156 -0
  153. package/dist/services/team-store.js.map +1 -0
  154. package/dist/services/trigger-log-store.d.ts +46 -0
  155. package/dist/services/trigger-log-store.d.ts.map +1 -0
  156. package/dist/services/trigger-log-store.js +132 -0
  157. package/dist/services/trigger-log-store.js.map +1 -0
  158. package/dist/services/trigger-types.d.ts +57 -0
  159. package/dist/services/trigger-types.d.ts.map +1 -0
  160. package/dist/services/trigger-types.js +28 -0
  161. package/dist/services/trigger-types.js.map +1 -0
  162. package/dist/services/webhook-key.d.ts +16 -0
  163. package/dist/services/webhook-key.d.ts.map +1 -0
  164. package/dist/services/webhook-key.js +123 -0
  165. package/dist/services/webhook-key.js.map +1 -0
  166. package/dist/services/webhook-lifecycle-extractors.d.ts +15 -0
  167. package/dist/services/webhook-lifecycle-extractors.d.ts.map +1 -0
  168. package/dist/services/webhook-lifecycle-extractors.js +59 -0
  169. package/dist/services/webhook-lifecycle-extractors.js.map +1 -0
  170. package/dist/services/webhook-lifecycle-store.d.ts +45 -0
  171. package/dist/services/webhook-lifecycle-store.d.ts.map +1 -0
  172. package/dist/services/webhook-lifecycle-store.js +159 -0
  173. package/dist/services/webhook-lifecycle-store.js.map +1 -0
  174. package/dist/setup/verify-permissions.d.ts.map +1 -1
  175. package/dist/setup/verify-permissions.js +4 -0
  176. package/dist/setup/verify-permissions.js.map +1 -1
  177. package/dist/skills/definitions.d.ts.map +1 -1
  178. package/dist/skills/definitions.js +64 -10
  179. package/dist/skills/definitions.js.map +1 -1
  180. package/dist/types.d.ts +14 -0
  181. package/dist/types.d.ts.map +1 -1
  182. package/dist/utils/daemon-discovery.d.ts +11 -0
  183. package/dist/utils/daemon-discovery.d.ts.map +1 -0
  184. package/dist/utils/daemon-discovery.js +59 -0
  185. package/dist/utils/daemon-discovery.js.map +1 -0
  186. package/dist/workflows/events/payloads.d.ts +2 -2
  187. package/dist/workflows/events/schema.d.ts +8 -8
  188. package/dist/workflows/trigger-from-envelope.d.ts +13 -0
  189. package/dist/workflows/trigger-from-envelope.d.ts.map +1 -0
  190. package/dist/workflows/trigger-from-envelope.js +67 -0
  191. package/dist/workflows/trigger-from-envelope.js.map +1 -0
  192. 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":"AAoBA,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAEhE,MAAM,CAAC,KAAK,UAAU,yBAAyB,CAC7C,cAA0C,EAC1C,OAAe,EACf,aAAqB,EACrB,cAAsB;IAEtB,MAAM,UAAU,GAAoB,EAAE,CAAC;IACvC,KAAK,MAAM,CAAC,IAAI,cAAc,CAAC,MAAM,EAAE,EAAE,CAAC;QACxC,IAAI,CAAC,CAAC,SAAS,KAAK,OAAO;YAAE,SAAS;QACtC,IAAI,CAAC,CAAC,MAAM,KAAK,aAAa;YAAE,SAAS;QACzC,IAAI,CAAC,CAAC,OAAO,CAAC,WAAW,KAAK,cAAc;YAAE,SAAS;QACvD,IAAI,CAAC,CAAC,OAAO,CAAC,WAAW;YAAE,SAAS;QACpC,oEAAoE;QACpE,oEAAoE;QACpE,kEAAkE;QAClE,mBAAmB;QACnB,IAAI,CAAC,sBAAsB,CAAC,CAAC,CAAC;YAAE,SAAS;QACzC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACrB,CAAC;IACD,yEAAyE;IACzE,yEAAyE;IACzE,0EAA0E;IAC1E,oEAAoE;IACpE,sEAAsE;IACtE,6DAA6D;IAC7D,sEAAsE;IACtE,sEAAsE;IACtE,MAAM,YAAY,GAAG,CAAC,GAAG,IAAI,GAAG,CAC9B,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAChE,CAAC,CAAC;IACH,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,CAChC,YAAY,CAAC,GAAG,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,EAAE,MAAM,kBAAkB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAU,CAAC,CACxF,CAAC;IACF,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAoE,CAAC;IAC7F,KAAK,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,QAAQ;QAAE,QAAQ,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IAC5D,OAAO,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;QACxB,IAAI,CAAC,CAAC,QAAQ,KAAK,KAAK,EAAE,CAAC;YACzB,OAAO;gBACL,SAAS,EAAE,CAAC,CAAC,OAAO,CAAC,SAAS;gBAC9B,+DAA+D;gBAC/D,8DAA8D;gBAC9D,8DAA8D;gBAC9D,SAAS,EAAE,CAAC,CAAC,MAAM;gBACnB,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,IAAI,CAAC,CAAC,gBAAgB,IAAI,YAAY;gBAC5D,UAAU,EAAE,CAAC,CAAC,OAAO,CAAC,UAAU;gBAChC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK;gBACtB,aAAa,EAAE,CAAC,CAAC,aAAa;gBAC9B,QAAQ,EAAE,KAAc;aACzB,CAAC;QACJ,CAAC;QACD,MAAM,IAAI,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QACpC,OAAO;YACL,SAAS,EAAE,CAAC,CAAC,OAAO,CAAC,SAAS;YAC9B,SAAS,EAAE,IAAI,EAAE,IAAI,IAAI,CAAC,CAAC,MAAM;YACjC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,IAAI,CAAC,CAAC,gBAAgB,IAAI,YAAY;YAC5D,UAAU,EAAE,CAAC,CAAC,OAAO,CAAC,UAAU;YAChC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK;YACtB,aAAa,EAAE,CAAC,CAAC,aAAa;YAC9B,QAAQ,EAAE,IAAI,EAAE,IAAI,IAAI,OAAO;SAChC,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -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"}