opencode-plugin-teleprompt 0.1.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 (54) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +180 -0
  3. package/dist/config.d.ts +2 -0
  4. package/dist/config.js +40 -0
  5. package/dist/config.js.map +1 -0
  6. package/dist/index.d.ts +7 -0
  7. package/dist/index.js +8 -0
  8. package/dist/index.js.map +1 -0
  9. package/dist/opencode/binding.d.ts +2 -0
  10. package/dist/opencode/binding.js +12 -0
  11. package/dist/opencode/binding.js.map +1 -0
  12. package/dist/opencode/events.d.ts +31 -0
  13. package/dist/opencode/events.js +74 -0
  14. package/dist/opencode/events.js.map +1 -0
  15. package/dist/opencode/permissions.d.ts +4 -0
  16. package/dist/opencode/permissions.js +66 -0
  17. package/dist/opencode/permissions.js.map +1 -0
  18. package/dist/opencode/submit.d.ts +7 -0
  19. package/dist/opencode/submit.js +14 -0
  20. package/dist/opencode/submit.js.map +1 -0
  21. package/dist/runtime/controller.d.ts +78 -0
  22. package/dist/runtime/controller.js +1180 -0
  23. package/dist/runtime/controller.js.map +1 -0
  24. package/dist/runtime/shutdown.d.ts +1 -0
  25. package/dist/runtime/shutdown.js +10 -0
  26. package/dist/runtime/shutdown.js.map +1 -0
  27. package/dist/state/lease.d.ts +12 -0
  28. package/dist/state/lease.js +56 -0
  29. package/dist/state/lease.js.map +1 -0
  30. package/dist/state/store.d.ts +9 -0
  31. package/dist/state/store.js +59 -0
  32. package/dist/state/store.js.map +1 -0
  33. package/dist/summary/format.d.ts +2 -0
  34. package/dist/summary/format.js +23 -0
  35. package/dist/summary/format.js.map +1 -0
  36. package/dist/telegram/api.d.ts +16 -0
  37. package/dist/telegram/api.js +78 -0
  38. package/dist/telegram/api.js.map +1 -0
  39. package/dist/telegram/parser.d.ts +3 -0
  40. package/dist/telegram/parser.js +267 -0
  41. package/dist/telegram/parser.js.map +1 -0
  42. package/dist/telegram/poller.d.ts +18 -0
  43. package/dist/telegram/poller.js +50 -0
  44. package/dist/telegram/poller.js.map +1 -0
  45. package/dist/tui-types.d.ts +55 -0
  46. package/dist/tui-types.js +2 -0
  47. package/dist/tui-types.js.map +1 -0
  48. package/dist/tui.d.ts +2 -0
  49. package/dist/tui.js +98 -0
  50. package/dist/tui.js.map +1 -0
  51. package/dist/types.d.ts +167 -0
  52. package/dist/types.js +2 -0
  53. package/dist/types.js.map +1 -0
  54. package/package.json +55 -0
@@ -0,0 +1,267 @@
1
+ function parseCommandBody(body) {
2
+ const trimmed = body.trim();
3
+ if (!trimmed)
4
+ return undefined;
5
+ const statusMatch = /^status\s*$/i.exec(trimmed);
6
+ if (statusMatch)
7
+ return { kind: "status" };
8
+ const interruptMatch = /^interrupt\s*$/i.exec(trimmed);
9
+ if (interruptMatch)
10
+ return { kind: "interrupt" };
11
+ const queueMatch = /^queue\s*$/i.exec(trimmed);
12
+ if (queueMatch)
13
+ return { kind: "queue" };
14
+ const retryMatch = /^retry\s*$/i.exec(trimmed);
15
+ if (retryMatch)
16
+ return { kind: "retry" };
17
+ const contextMatch = /^context\s*$/i.exec(trimmed);
18
+ if (contextMatch)
19
+ return { kind: "context" };
20
+ const compactMatch = /^compact\s*$/i.exec(trimmed);
21
+ if (compactMatch)
22
+ return { kind: "compact" };
23
+ const newSessionMatch = /^(newsession|new-session)\s*$/i.exec(trimmed);
24
+ if (newSessionMatch)
25
+ return { kind: "newsession" };
26
+ const resetContextMatch = /^reset-context\s*$/i.exec(trimmed);
27
+ if (resetContextMatch)
28
+ return { kind: "reset-context" };
29
+ const whoMatch = /^who\s*$/i.exec(trimmed);
30
+ if (whoMatch)
31
+ return { kind: "who" };
32
+ const healthMatch = /^health\s*$/i.exec(trimmed);
33
+ if (healthMatch)
34
+ return { kind: "health" };
35
+ const reclaimMatch = /^reclaim\s*$/i.exec(trimmed);
36
+ if (reclaimMatch)
37
+ return { kind: "reclaim" };
38
+ const historyMatch = /^history\s*$/i.exec(trimmed);
39
+ if (historyMatch)
40
+ return { kind: "history" };
41
+ const lastErrorMatch = /^last-error\s*$/i.exec(trimmed);
42
+ if (lastErrorMatch)
43
+ return { kind: "last-error" };
44
+ const cancelMatch = /^cancel\s+([A-Za-z0-9_\-:.]+)\s*$/i.exec(trimmed);
45
+ if (cancelMatch)
46
+ return { kind: "cancel", target: cancelMatch[1] };
47
+ const approveMatch = /^approve\s+([A-Za-z0-9_\-:.]+)\s*$/i.exec(trimmed);
48
+ if (approveMatch) {
49
+ return { kind: "permission", action: "once", requestID: approveMatch[1] };
50
+ }
51
+ const approveAlwaysMatch = /^approve-always\s+([A-Za-z0-9_\-:.]+)\s*$/i.exec(trimmed);
52
+ if (approveAlwaysMatch) {
53
+ return {
54
+ kind: "permission",
55
+ action: "always",
56
+ requestID: approveAlwaysMatch[1],
57
+ };
58
+ }
59
+ const denyMatch = /^deny\s+([A-Za-z0-9_\-:.]+)\s*$/i.exec(trimmed);
60
+ if (denyMatch) {
61
+ return { kind: "permission", action: "reject", requestID: denyMatch[1] };
62
+ }
63
+ return { kind: "prompt", prompt: trimmed };
64
+ }
65
+ export function parseTelegramUpdate(update, channelID, prefix) {
66
+ const post = update.channel_post;
67
+ if (!post)
68
+ return undefined;
69
+ const normalizedChannel = String(post.chat.id);
70
+ if (normalizedChannel !== channelID)
71
+ return undefined;
72
+ const text = post.text?.trim();
73
+ if (!text)
74
+ return undefined;
75
+ if (text.startsWith(`${prefix}:`)) {
76
+ const colonBody = text.slice(`${prefix}:`.length).trim();
77
+ if (!colonBody)
78
+ return undefined;
79
+ if (/^dc\s*$/i.test(colonBody)) {
80
+ return {
81
+ updateID: update.update_id,
82
+ messageID: post.message_id,
83
+ channelID: normalizedChannel,
84
+ rawText: text,
85
+ command: { kind: "disconnect" },
86
+ };
87
+ }
88
+ if (/^interrupt\s*$/i.test(colonBody)) {
89
+ return {
90
+ updateID: update.update_id,
91
+ messageID: post.message_id,
92
+ channelID: normalizedChannel,
93
+ rawText: text,
94
+ command: { kind: "interrupt" },
95
+ };
96
+ }
97
+ if (/^queue\s*$/i.test(colonBody)) {
98
+ return {
99
+ updateID: update.update_id,
100
+ messageID: post.message_id,
101
+ channelID: normalizedChannel,
102
+ rawText: text,
103
+ command: { kind: "queue" },
104
+ };
105
+ }
106
+ if (/^retry\s*$/i.test(colonBody)) {
107
+ return {
108
+ updateID: update.update_id,
109
+ messageID: post.message_id,
110
+ channelID: normalizedChannel,
111
+ rawText: text,
112
+ command: { kind: "retry" },
113
+ };
114
+ }
115
+ if (/^context\s*$/i.test(colonBody)) {
116
+ return {
117
+ updateID: update.update_id,
118
+ messageID: post.message_id,
119
+ channelID: normalizedChannel,
120
+ rawText: text,
121
+ command: { kind: "context" },
122
+ };
123
+ }
124
+ if (/^compact\s*$/i.test(colonBody)) {
125
+ return {
126
+ updateID: update.update_id,
127
+ messageID: post.message_id,
128
+ channelID: normalizedChannel,
129
+ rawText: text,
130
+ command: { kind: "compact" },
131
+ };
132
+ }
133
+ if (/^(newsession|new-session)\s*$/i.test(colonBody)) {
134
+ return {
135
+ updateID: update.update_id,
136
+ messageID: post.message_id,
137
+ channelID: normalizedChannel,
138
+ rawText: text,
139
+ command: { kind: "newsession" },
140
+ };
141
+ }
142
+ if (/^reset-context\s*$/i.test(colonBody)) {
143
+ return {
144
+ updateID: update.update_id,
145
+ messageID: post.message_id,
146
+ channelID: normalizedChannel,
147
+ rawText: text,
148
+ command: { kind: "reset-context" },
149
+ };
150
+ }
151
+ if (/^who\s*$/i.test(colonBody)) {
152
+ return {
153
+ updateID: update.update_id,
154
+ messageID: post.message_id,
155
+ channelID: normalizedChannel,
156
+ rawText: text,
157
+ command: { kind: "who" },
158
+ };
159
+ }
160
+ if (/^health\s*$/i.test(colonBody)) {
161
+ return {
162
+ updateID: update.update_id,
163
+ messageID: post.message_id,
164
+ channelID: normalizedChannel,
165
+ rawText: text,
166
+ command: { kind: "health" },
167
+ };
168
+ }
169
+ if (/^reclaim\s*$/i.test(colonBody)) {
170
+ return {
171
+ updateID: update.update_id,
172
+ messageID: post.message_id,
173
+ channelID: normalizedChannel,
174
+ rawText: text,
175
+ command: { kind: "reclaim" },
176
+ };
177
+ }
178
+ if (/^history\s*$/i.test(colonBody)) {
179
+ return {
180
+ updateID: update.update_id,
181
+ messageID: post.message_id,
182
+ channelID: normalizedChannel,
183
+ rawText: text,
184
+ command: { kind: "history" },
185
+ };
186
+ }
187
+ if (/^last-error\s*$/i.test(colonBody)) {
188
+ return {
189
+ updateID: update.update_id,
190
+ messageID: post.message_id,
191
+ channelID: normalizedChannel,
192
+ rawText: text,
193
+ command: { kind: "last-error" },
194
+ };
195
+ }
196
+ const cancelMatch = /^cancel\s+([A-Za-z0-9_\-:.]+)\s*$/i.exec(colonBody);
197
+ if (cancelMatch) {
198
+ return {
199
+ updateID: update.update_id,
200
+ messageID: post.message_id,
201
+ channelID: normalizedChannel,
202
+ rawText: text,
203
+ command: { kind: "cancel", target: cancelMatch[1] },
204
+ };
205
+ }
206
+ const modelMatch = /^model(?:\s+([A-Za-z0-9_./-]+))?\s*$/i.exec(colonBody);
207
+ if (modelMatch) {
208
+ const rawTarget = modelMatch[1]?.trim();
209
+ if (!rawTarget) {
210
+ return {
211
+ updateID: update.update_id,
212
+ messageID: post.message_id,
213
+ channelID: normalizedChannel,
214
+ rawText: text,
215
+ command: { kind: "model" },
216
+ };
217
+ }
218
+ const preset = rawTarget.toLowerCase();
219
+ if (preset === "fast" || preset === "smart" || preset === "max") {
220
+ return {
221
+ updateID: update.update_id,
222
+ messageID: post.message_id,
223
+ channelID: normalizedChannel,
224
+ rawText: text,
225
+ command: { kind: "model", preset },
226
+ };
227
+ }
228
+ const providerModel = /^([A-Za-z0-9_.-]+)\/([A-Za-z0-9_.:-]+)$/.exec(rawTarget);
229
+ if (providerModel) {
230
+ return {
231
+ updateID: update.update_id,
232
+ messageID: post.message_id,
233
+ channelID: normalizedChannel,
234
+ rawText: text,
235
+ command: {
236
+ kind: "model",
237
+ target: {
238
+ providerID: providerModel[1],
239
+ modelID: providerModel[2],
240
+ },
241
+ },
242
+ };
243
+ }
244
+ return undefined;
245
+ }
246
+ return undefined;
247
+ }
248
+ if (!text.startsWith(prefix))
249
+ return undefined;
250
+ const remainder = text.slice(prefix.length).trim();
251
+ const command = parseCommandBody(remainder);
252
+ if (!command)
253
+ return undefined;
254
+ return {
255
+ updateID: update.update_id,
256
+ messageID: post.message_id,
257
+ channelID: normalizedChannel,
258
+ rawText: text,
259
+ command,
260
+ };
261
+ }
262
+ export function isChannelPostFromTarget(post, channelID) {
263
+ if (!post)
264
+ return false;
265
+ return String(post.chat.id) === channelID;
266
+ }
267
+ //# sourceMappingURL=parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parser.js","sourceRoot":"","sources":["../../src/telegram/parser.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { ParsedTelegramCommand } from "../types.js";
2
+ import { TelegramApi } from "./api.js";
3
+ type PollerHandlers = {
4
+ onCommand: (command: ParsedTelegramCommand) => Promise<void>;
5
+ onOffset: (offset: number) => Promise<void>;
6
+ onError: (error: unknown) => void;
7
+ };
8
+ export declare class TelegramPoller {
9
+ private readonly api;
10
+ private readonly channelID;
11
+ private readonly prefix;
12
+ private readonly timeoutSec;
13
+ private readonly handlers;
14
+ constructor(api: TelegramApi, channelID: string, prefix: string, timeoutSec: number, handlers: PollerHandlers);
15
+ run(startOffset: number, signal: AbortSignal): Promise<void>;
16
+ private processUpdates;
17
+ }
18
+ export {};
@@ -0,0 +1,50 @@
1
+ import { parseTelegramUpdate } from "./parser.js";
2
+ export class TelegramPoller {
3
+ api;
4
+ channelID;
5
+ prefix;
6
+ timeoutSec;
7
+ handlers;
8
+ constructor(api, channelID, prefix, timeoutSec, handlers) {
9
+ this.api = api;
10
+ this.channelID = channelID;
11
+ this.prefix = prefix;
12
+ this.timeoutSec = timeoutSec;
13
+ this.handlers = handlers;
14
+ }
15
+ async run(startOffset, signal) {
16
+ let offset = startOffset;
17
+ while (!signal.aborted) {
18
+ try {
19
+ const updates = await this.api.getUpdates(offset, this.timeoutSec, signal);
20
+ if (signal.aborted)
21
+ break;
22
+ offset = await this.processUpdates(offset, updates);
23
+ }
24
+ catch (error) {
25
+ if (signal.aborted)
26
+ break;
27
+ this.handlers.onError(error);
28
+ await new Promise((resolve) => setTimeout(resolve, 1000));
29
+ }
30
+ }
31
+ }
32
+ async processUpdates(offset, updates) {
33
+ let nextOffset = offset;
34
+ for (const update of updates) {
35
+ const parsed = parseTelegramUpdate(update, this.channelID, this.prefix);
36
+ const candidateOffset = Math.max(nextOffset, update.update_id + 1);
37
+ if (!parsed) {
38
+ nextOffset = candidateOffset;
39
+ await this.handlers.onOffset(nextOffset);
40
+ continue;
41
+ }
42
+ // Persist offset only after command handling succeeds, so failed commands are retried.
43
+ await this.handlers.onCommand(parsed);
44
+ nextOffset = candidateOffset;
45
+ await this.handlers.onOffset(nextOffset);
46
+ }
47
+ return nextOffset;
48
+ }
49
+ }
50
+ //# sourceMappingURL=poller.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"poller.js","sourceRoot":"","sources":["../../src/telegram/poller.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,aAAa,CAAC;AASlD,MAAM,OAAO,cAAc;IAEN;IACA;IACA;IACA;IACA;IALnB,YACmB,GAAgB,EAChB,SAAiB,EACjB,MAAc,EACd,UAAkB,EAClB,QAAwB;QAJxB,QAAG,GAAH,GAAG,CAAa;QAChB,cAAS,GAAT,SAAS,CAAQ;QACjB,WAAM,GAAN,MAAM,CAAQ;QACd,eAAU,GAAV,UAAU,CAAQ;QAClB,aAAQ,GAAR,QAAQ,CAAgB;IACxC,CAAC;IAEJ,KAAK,CAAC,GAAG,CAAC,WAAmB,EAAE,MAAmB;QAChD,IAAI,MAAM,GAAG,WAAW,CAAC;QACzB,OAAO,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACvB,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,MAAM,EAAE,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;gBAC3E,IAAI,MAAM,CAAC,OAAO;oBAAE,MAAM;gBAC1B,MAAM,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YACtD,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,IAAI,MAAM,CAAC,OAAO;oBAAE,MAAM;gBAC1B,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;gBAC7B,MAAM,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,cAAc,CAC1B,MAAc,EACd,OAAyB;QAEzB,IAAI,UAAU,GAAG,MAAM,CAAC;QACxB,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,MAAM,MAAM,GAAG,mBAAmB,CAAC,MAAM,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;YACxE,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,MAAM,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;YACnE,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,UAAU,GAAG,eAAe,CAAC;gBAC7B,MAAM,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBACzC,SAAS;YACX,CAAC;YACD,uFAAuF;YACvF,MAAM,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;YACtC,UAAU,GAAG,eAAe,CAAC;YAC7B,MAAM,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;QAC3C,CAAC;QACD,OAAO,UAAU,CAAC;IACpB,CAAC;CACF"}
@@ -0,0 +1,55 @@
1
+ export type TuiCommand = {
2
+ title: string;
3
+ value: string;
4
+ description?: string;
5
+ category?: string;
6
+ slash?: {
7
+ name: string;
8
+ aliases?: string[];
9
+ };
10
+ onSelect?: () => void;
11
+ };
12
+ export type TuiPluginApi = {
13
+ command: {
14
+ register: (cb: () => TuiCommand[]) => () => void;
15
+ };
16
+ route: {
17
+ current: {
18
+ name: "session";
19
+ params: {
20
+ sessionID?: string;
21
+ };
22
+ } | {
23
+ name: string;
24
+ params?: Record<string, unknown>;
25
+ };
26
+ };
27
+ ui: {
28
+ toast: (input: {
29
+ variant?: "info" | "success" | "warning" | "error";
30
+ message: string;
31
+ }) => void;
32
+ };
33
+ lifecycle: {
34
+ onDispose: (fn: () => void | Promise<void>) => () => void;
35
+ };
36
+ event: {
37
+ on: (type: "tui.command.execute", handler: (event: {
38
+ type: "tui.command.execute";
39
+ properties: {
40
+ command: string;
41
+ };
42
+ }) => void) => () => void;
43
+ };
44
+ state: {
45
+ path: {
46
+ directory: string;
47
+ };
48
+ part: (messageID: string) => ReadonlyArray<{
49
+ type: string;
50
+ [key: string]: unknown;
51
+ }>;
52
+ };
53
+ client: any;
54
+ };
55
+ export type TuiPlugin = (api: TuiPluginApi, options?: Record<string, unknown>) => Promise<void>;
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=tui-types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tui-types.js","sourceRoot":"","sources":["../src/tui-types.ts"],"names":[],"mappings":""}
package/dist/tui.d.ts ADDED
@@ -0,0 +1,2 @@
1
+ import type { TuiPlugin } from "./tui-types.js";
2
+ export declare const tui: TuiPlugin;
package/dist/tui.js ADDED
@@ -0,0 +1,98 @@
1
+ import { join } from "node:path";
2
+ import { loadConfig } from "./config.js";
3
+ import { BridgeController } from "./runtime/controller.js";
4
+ async function runSafe(api, fn) {
5
+ try {
6
+ const message = await fn();
7
+ if (message) {
8
+ api.ui.toast({
9
+ variant: "success",
10
+ message,
11
+ });
12
+ }
13
+ }
14
+ catch (error) {
15
+ api.ui.toast({
16
+ variant: "error",
17
+ message: String(error),
18
+ });
19
+ }
20
+ }
21
+ export const tui = async (api, _options) => {
22
+ const config = loadConfig();
23
+ const storePath = join(api.state.path.directory, ".opencode-telegram-bridge.json");
24
+ const controller = new BridgeController(api, config, storePath);
25
+ await controller.init();
26
+ const unregister = api.command.register(() => [
27
+ {
28
+ title: "Teleprompt: Start",
29
+ value: "tp:start",
30
+ description: "Activate Telegram bridge for current OpenCode session",
31
+ category: "Teleprompt",
32
+ slash: {
33
+ name: "tp:start",
34
+ aliases: ["telegram.bind"],
35
+ },
36
+ onSelect: () => {
37
+ void runSafe(api, async () => {
38
+ const sessionID = await controller.bindCurrent();
39
+ return `Telegram bridge bound to session ${sessionID}`;
40
+ });
41
+ },
42
+ },
43
+ {
44
+ title: "Teleprompt: Stop",
45
+ value: "tp:stop",
46
+ description: "Disconnect Telegram bridge and unlock local input",
47
+ category: "Teleprompt",
48
+ slash: {
49
+ name: "tp:stop",
50
+ aliases: ["telegram.unbind"],
51
+ },
52
+ onSelect: () => {
53
+ void runSafe(api, async () => {
54
+ await controller.unbind();
55
+ return "Telegram bridge unbound";
56
+ });
57
+ },
58
+ },
59
+ {
60
+ title: "Teleprompt: Status",
61
+ value: "tp:status",
62
+ description: "Show bridge status",
63
+ category: "Teleprompt",
64
+ slash: {
65
+ name: "tp:status",
66
+ aliases: ["telegram.status"],
67
+ },
68
+ onSelect: () => {
69
+ void runSafe(api, async () => controller.statusLine());
70
+ },
71
+ },
72
+ {
73
+ title: "Teleprompt: Credentials",
74
+ value: "tp:credentials",
75
+ description: "Set session-only Telegram credentials",
76
+ category: "Teleprompt",
77
+ slash: {
78
+ name: "tp:credentials",
79
+ aliases: ["telegram.credentials"],
80
+ },
81
+ onSelect: () => {
82
+ api.ui.toast({
83
+ variant: "info",
84
+ message: "Usage: /tp:credentials <bot_token> <channel_id> then /tp:start",
85
+ });
86
+ },
87
+ },
88
+ ]);
89
+ const unsubscribeTuiCommand = api.event.on("tui.command.execute", (event) => {
90
+ void controller.handleLocalTuiCommand(event.properties.command);
91
+ });
92
+ api.lifecycle.onDispose(async () => {
93
+ unsubscribeTuiCommand();
94
+ unregister();
95
+ await controller.shutdown();
96
+ });
97
+ };
98
+ //# sourceMappingURL=tui.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tui.js","sourceRoot":"","sources":["../src/tui.ts"],"names":[],"mappings":"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"}