ccbot 1.0.0 → 1.2.1

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 (79) hide show
  1. package/dist/channel/telegram/telegram-channel.d.ts +18 -0
  2. package/dist/channel/telegram/telegram-channel.js +148 -0
  3. package/dist/channel/telegram/telegram-channel.js.map +1 -0
  4. package/dist/channel/telegram/telegram-sender.d.ts +2 -0
  5. package/dist/channel/telegram/telegram-sender.js +72 -0
  6. package/dist/channel/telegram/telegram-sender.js.map +1 -0
  7. package/dist/channel/types.d.ts +20 -0
  8. package/dist/channel/types.js +2 -0
  9. package/dist/channel/types.js.map +1 -0
  10. package/dist/commands/help.js +11 -9
  11. package/dist/commands/help.js.map +1 -0
  12. package/dist/commands/setup.js +49 -40
  13. package/dist/commands/setup.js.map +1 -0
  14. package/dist/commands/uninstall.js +15 -13
  15. package/dist/commands/uninstall.js.map +1 -0
  16. package/dist/commands/update.js +90 -39
  17. package/dist/commands/update.js.map +1 -0
  18. package/dist/config-manager.d.ts +7 -2
  19. package/dist/config-manager.js +44 -21
  20. package/dist/config-manager.js.map +1 -0
  21. package/dist/hook/hook-handler.d.ts +8 -7
  22. package/dist/hook/hook-handler.js +53 -100
  23. package/dist/hook/hook-handler.js.map +1 -0
  24. package/dist/hook/hook-installer.d.ts +1 -3
  25. package/dist/hook/hook-installer.js +46 -33
  26. package/dist/hook/hook-installer.js.map +1 -0
  27. package/dist/hook/hook-server.d.ts +2 -1
  28. package/dist/hook/hook-server.js +32 -9
  29. package/dist/hook/hook-server.js.map +1 -0
  30. package/dist/hook/response-store.d.ts +8 -0
  31. package/dist/hook/response-store.js +18 -0
  32. package/dist/i18n/index.d.ts +9 -0
  33. package/dist/i18n/index.js +55 -0
  34. package/dist/i18n/index.js.map +1 -0
  35. package/dist/i18n/locales/en.d.ts +2 -0
  36. package/dist/i18n/locales/en.js +127 -0
  37. package/dist/i18n/locales/en.js.map +1 -0
  38. package/dist/i18n/locales/vi.d.ts +2 -0
  39. package/dist/i18n/locales/vi.js +127 -0
  40. package/dist/i18n/locales/vi.js.map +1 -0
  41. package/dist/i18n/locales/zh.d.ts +2 -0
  42. package/dist/i18n/locales/zh.js +127 -0
  43. package/dist/i18n/locales/zh.js.map +1 -0
  44. package/dist/i18n/types.d.ts +126 -0
  45. package/dist/i18n/types.js +2 -0
  46. package/dist/i18n/types.js.map +1 -0
  47. package/dist/index.js +39 -18
  48. package/dist/index.js.map +1 -0
  49. package/dist/monitor/git-collector.d.ts +2 -0
  50. package/dist/monitor/git-collector.js +88 -0
  51. package/dist/monitor/git-collector.js.map +1 -0
  52. package/dist/monitor/transcript-parser.d.ts +4 -0
  53. package/dist/monitor/transcript-parser.js +35 -14
  54. package/dist/monitor/transcript-parser.js.map +1 -0
  55. package/dist/telegram/bot.d.ts +8 -3
  56. package/dist/telegram/bot.js +69 -18
  57. package/dist/telegram/message-formatter.d.ts +7 -2
  58. package/dist/telegram/message-formatter.js +26 -23
  59. package/dist/telegram/message-sender.d.ts +1 -1
  60. package/dist/telegram/message-sender.js +37 -5
  61. package/dist/utils/constants.d.ts +44 -0
  62. package/dist/utils/constants.js +41 -0
  63. package/dist/utils/constants.js.map +1 -0
  64. package/dist/utils/install-detection.d.ts +2 -1
  65. package/dist/utils/install-detection.js +11 -9
  66. package/dist/utils/install-detection.js.map +1 -0
  67. package/dist/utils/log.d.ts +2 -0
  68. package/dist/utils/log.js +18 -0
  69. package/dist/utils/log.js.map +1 -0
  70. package/dist/utils/paths.d.ts +16 -0
  71. package/dist/utils/paths.js +39 -0
  72. package/dist/utils/paths.js.map +1 -0
  73. package/dist/utils/response-store.d.ts +29 -0
  74. package/dist/utils/response-store.js +100 -0
  75. package/dist/utils/response-store.js.map +1 -0
  76. package/dist/utils/tunnel.d.ts +7 -0
  77. package/dist/utils/tunnel.js +53 -0
  78. package/dist/utils/tunnel.js.map +1 -0
  79. package/package.json +30 -12
@@ -0,0 +1,18 @@
1
+ import type { Config } from "../../config-manager.js";
2
+ import type { NotificationChannel, NotificationData } from "../types.js";
3
+ export declare class TelegramChannel implements NotificationChannel {
4
+ private bot;
5
+ private cfg;
6
+ private chatId;
7
+ private isDisconnected;
8
+ private tunnelUrl;
9
+ constructor(cfg: Config, tunnelUrl?: string | null);
10
+ initialize(): Promise<void>;
11
+ shutdown(): Promise<void>;
12
+ sendNotification(data: NotificationData, responseUrl?: string): Promise<void>;
13
+ private formatNotification;
14
+ private registerCommands;
15
+ private registerMenuButton;
16
+ private registerHandlers;
17
+ private registerPollingErrorHandler;
18
+ }
@@ -0,0 +1,148 @@
1
+ import TelegramBot from "node-telegram-bot-api";
2
+ import { ConfigManager } from "../../config-manager.js";
3
+ import { sendTelegramMessage } from "./telegram-sender.js";
4
+ import { MINI_APP_BASE_URL } from "../../utils/constants.js";
5
+ import { t, getTranslations } from "../../i18n/index.js";
6
+ import { log, logError } from "../../utils/log.js";
7
+ export class TelegramChannel {
8
+ bot;
9
+ cfg;
10
+ chatId = null;
11
+ isDisconnected = false;
12
+ tunnelUrl;
13
+ constructor(cfg, tunnelUrl) {
14
+ this.cfg = cfg;
15
+ this.tunnelUrl = tunnelUrl ?? null;
16
+ this.bot = new TelegramBot(cfg.telegram_bot_token, { polling: false });
17
+ this.chatId = ConfigManager.loadChatState().chat_id;
18
+ this.registerHandlers();
19
+ this.registerPollingErrorHandler();
20
+ }
21
+ async initialize() {
22
+ this.bot.startPolling();
23
+ await this.registerCommands();
24
+ await this.registerMenuButton();
25
+ log(t("bot.telegramStarted"));
26
+ }
27
+ async shutdown() {
28
+ this.bot.stopPolling();
29
+ }
30
+ async sendNotification(data, responseUrl) {
31
+ if (!this.chatId) {
32
+ log(t("bot.noChatId"));
33
+ return;
34
+ }
35
+ const text = this.formatNotification(data);
36
+ try {
37
+ await sendTelegramMessage(this.bot, this.chatId, text, responseUrl);
38
+ }
39
+ catch (err) {
40
+ logError(t("bot.notificationFailed"), err);
41
+ }
42
+ }
43
+ formatNotification(data) {
44
+ const parts = [];
45
+ let header = `<b>${escapeHtml(data.projectName)}</b>`;
46
+ if (data.durationMs > 0) {
47
+ header += ` — ${escapeHtml(formatDuration(data.durationMs))}`;
48
+ }
49
+ parts.push(header);
50
+ if (data.inputTokens > 0 || data.outputTokens > 0) {
51
+ parts.push(`${t("notification.tokens")}: ${formatTokenCount(data.inputTokens)} → ${formatTokenCount(data.outputTokens)}`);
52
+ }
53
+ if (data.cacheReadTokens > 0 || data.cacheCreationTokens > 0) {
54
+ const cacheParts = [];
55
+ if (data.cacheReadTokens > 0)
56
+ cacheParts.push(`${formatTokenCount(data.cacheReadTokens)} ${t("notification.cacheRead")}`);
57
+ if (data.cacheCreationTokens > 0)
58
+ cacheParts.push(`${formatTokenCount(data.cacheCreationTokens)} ${t("notification.cacheWrite")}`);
59
+ parts.push(`${t("notification.cache")}: ${cacheParts.join(", ")}`);
60
+ }
61
+ return parts.join("\n");
62
+ }
63
+ async registerCommands() {
64
+ const translations = getTranslations();
65
+ const commands = [
66
+ { command: "start", description: translations.bot.commands.start },
67
+ ];
68
+ try {
69
+ await this.bot.setMyCommands(commands);
70
+ log(t("bot.commandsRegistered"));
71
+ }
72
+ catch (err) {
73
+ logError(t("bot.commandsRegisterFailed"), err);
74
+ }
75
+ }
76
+ async registerMenuButton() {
77
+ const url = this.tunnelUrl ? `${this.tunnelUrl}/` : `${MINI_APP_BASE_URL}/`;
78
+ try {
79
+ await this.bot.setChatMenuButton({
80
+ menu_button: JSON.stringify({
81
+ type: "web_app",
82
+ text: t("bot.dashboard"),
83
+ web_app: { url },
84
+ }),
85
+ });
86
+ log(t("bot.menuButtonRegistered"));
87
+ }
88
+ catch (err) {
89
+ logError(t("bot.menuButtonFailed"), err);
90
+ }
91
+ }
92
+ registerHandlers() {
93
+ this.bot.on("message", (msg) => {
94
+ if (!ConfigManager.isOwner(this.cfg, msg.from?.id ?? 0)) {
95
+ log(t("bot.unauthorizedUser", {
96
+ userId: msg.from?.id ?? 0,
97
+ username: msg.from?.username ?? "",
98
+ }));
99
+ return;
100
+ }
101
+ const text = msg.text ?? "";
102
+ if (text === "/start") {
103
+ if (this.chatId === msg.chat.id) {
104
+ this.bot.sendMessage(msg.chat.id, t("bot.alreadyConnected"));
105
+ return;
106
+ }
107
+ this.chatId = msg.chat.id;
108
+ ConfigManager.saveChatState({ chat_id: this.chatId });
109
+ log(t("bot.registeredChatId", { chatId: msg.chat.id }));
110
+ this.bot.sendMessage(msg.chat.id, t("bot.ready"), { parse_mode: "MarkdownV2" });
111
+ return;
112
+ }
113
+ });
114
+ }
115
+ registerPollingErrorHandler() {
116
+ this.bot.on("polling_error", () => {
117
+ if (!this.isDisconnected) {
118
+ this.isDisconnected = true;
119
+ log(t("bot.connectionLost"));
120
+ }
121
+ });
122
+ this.bot.on("polling", () => {
123
+ if (this.isDisconnected) {
124
+ this.isDisconnected = false;
125
+ log(t("bot.connectionRestored"));
126
+ }
127
+ });
128
+ }
129
+ }
130
+ function escapeHtml(text) {
131
+ return text.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
132
+ }
133
+ function formatDuration(ms) {
134
+ const totalSeconds = Math.floor(ms / 1000);
135
+ if (totalSeconds < 60)
136
+ return `${totalSeconds}s`;
137
+ const minutes = Math.floor(totalSeconds / 60);
138
+ const seconds = totalSeconds % 60;
139
+ return `${minutes}m${seconds}s`;
140
+ }
141
+ function formatTokenCount(tokens) {
142
+ if (tokens >= 1_000_000)
143
+ return `${(tokens / 1_000_000).toFixed(1)}M`;
144
+ if (tokens >= 1_000)
145
+ return `${(tokens / 1_000).toFixed(1)}k`;
146
+ return `${tokens}`;
147
+ }
148
+ //# sourceMappingURL=telegram-channel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-channel.js","sourceRoot":"","sources":["../../../src/channel/telegram/telegram-channel.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import TelegramBot from "node-telegram-bot-api";
2
+ export declare function sendTelegramMessage(bot: TelegramBot, chatId: number, text: string, responseUrl?: string): Promise<void>;
@@ -0,0 +1,72 @@
1
+ import TelegramBot from "node-telegram-bot-api";
2
+ import { t } from "../../i18n/index.js";
3
+ import { logError } from "../../utils/log.js";
4
+ import { SPLIT_LOOKBACK_RANGE } from "../../utils/constants.js";
5
+ const TELEGRAM_MAX_MESSAGE_LENGTH = 4096;
6
+ const PAGINATION_FOOTER_RESERVE = 30;
7
+ export async function sendTelegramMessage(bot, chatId, text, responseUrl) {
8
+ const pages = splitMessage(text, TELEGRAM_MAX_MESSAGE_LENGTH - PAGINATION_FOOTER_RESERVE);
9
+ for (let i = 0; i < pages.length; i++) {
10
+ let content = pages[i];
11
+ if (pages.length > 1) {
12
+ content = `${content}\n\n<i>[${i + 1}/${pages.length}]</i>`;
13
+ }
14
+ const isLastPage = i === pages.length - 1;
15
+ const opts = { parse_mode: "HTML" };
16
+ if (isLastPage && responseUrl) {
17
+ opts.reply_markup = buildResponseReplyMarkup(responseUrl);
18
+ }
19
+ try {
20
+ await bot.sendMessage(chatId, content, opts);
21
+ }
22
+ catch (err) {
23
+ logError(t("bot.sendFailed"), err);
24
+ const fallbackOpts = {};
25
+ if (isLastPage && responseUrl) {
26
+ fallbackOpts.reply_markup = buildResponseReplyMarkup(responseUrl);
27
+ }
28
+ try {
29
+ await bot.sendMessage(chatId, pages[i], fallbackOpts);
30
+ }
31
+ catch (err2) {
32
+ logError(t("bot.sendFallbackFailed"), err2);
33
+ await bot.sendMessage(chatId, pages[i]);
34
+ }
35
+ }
36
+ }
37
+ }
38
+ function splitMessage(text, maxLen) {
39
+ if (text.length <= maxLen)
40
+ return [text];
41
+ const pages = [];
42
+ let remaining = text;
43
+ while (remaining.length > 0) {
44
+ if (remaining.length <= maxLen) {
45
+ pages.push(remaining);
46
+ break;
47
+ }
48
+ const splitAt = findSplitPoint(remaining, maxLen);
49
+ pages.push(remaining.slice(0, splitAt));
50
+ remaining = remaining.slice(splitAt);
51
+ }
52
+ return pages;
53
+ }
54
+ function findSplitPoint(text, maxLen) {
55
+ for (let i = maxLen; i > maxLen - SPLIT_LOOKBACK_RANGE && i > 0; i--) {
56
+ if (text[i] === "\n" && text[i - 1] !== "\\")
57
+ return i + 1;
58
+ }
59
+ for (let i = maxLen; i > maxLen - SPLIT_LOOKBACK_RANGE && i > 0; i--) {
60
+ if (text[i] === " " && text[i - 1] !== "\\")
61
+ return i + 1;
62
+ }
63
+ return maxLen;
64
+ }
65
+ function buildResponseReplyMarkup(responseUrl) {
66
+ const buttonText = t("bot.viewDetails");
67
+ const button = responseUrl.startsWith("https://")
68
+ ? { text: buttonText, web_app: { url: responseUrl } }
69
+ : { text: buttonText, url: responseUrl };
70
+ return { inline_keyboard: [[button]] };
71
+ }
72
+ //# sourceMappingURL=telegram-sender.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"telegram-sender.js","sourceRoot":"","sources":["../../../src/channel/telegram/telegram-sender.ts"],"names":[],"mappings":"AAAA,OAAO,WAAW,MAAM,uBAAuB,CAAC;AAChD,OAAO,EAAE,CAAC,EAAE,MAAM,qBAAqB,CAAC;AACxC,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAEhE,MAAM,2BAA2B,GAAG,IAAI,CAAC;AACzC,MAAM,yBAAyB,GAAG,EAAE,CAAC;AAErC,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,GAAgB,EAChB,MAAc,EACd,IAAY,EACZ,WAAoB;IAEpB,MAAM,KAAK,GAAG,YAAY,CAAC,IAAI,EAAE,2BAA2B,GAAG,yBAAyB,CAAC,CAAC;IAE1F,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACtC,IAAI,OAAO,GAAG,KAAK,CAAC,CAAC,CAAE,CAAC;QACxB,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACrB,OAAO,GAAG,GAAG,OAAO,WAAW,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,MAAM,OAAO,CAAC;QAC9D,CAAC;QAED,MAAM,UAAU,GAAG,CAAC,KAAK,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;QAC1C,MAAM,IAAI,GAAmC,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC;QAEpE,IAAI,UAAU,IAAI,WAAW,EAAE,CAAC;YAC9B,IAAI,CAAC,YAAY,GAAG,wBAAwB,CAAC,WAAW,CAAC,CAAC;QAC5D,CAAC;QAED,IAAI,CAAC;YACH,MAAM,GAAG,CAAC,WAAW,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,QAAQ,CAAC,CAAC,CAAC,gBAAgB,CAAC,EAAE,GAAG,CAAC,CAAC;YACnC,MAAM,YAAY,GAAmC,EAAE,CAAC;YACxD,IAAI,UAAU,IAAI,WAAW,EAAE,CAAC;gBAC9B,YAAY,CAAC,YAAY,GAAG,wBAAwB,CAAC,WAAW,CAAC,CAAC;YACpE,CAAC;YACD,IAAI,CAAC;gBACH,MAAM,GAAG,CAAC,WAAW,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAE,EAAE,YAAY,CAAC,CAAC;YACzD,CAAC;YAAC,OAAO,IAAI,EAAE,CAAC;gBACd,QAAQ,CAAC,CAAC,CAAC,wBAAwB,CAAC,EAAE,IAAI,CAAC,CAAC;gBAC5C,MAAM,GAAG,CAAC,WAAW,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAE,CAAC,CAAC;YAC3C,CAAC;QACH,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,YAAY,CAAC,IAAY,EAAE,MAAc;IAChD,IAAI,IAAI,CAAC,MAAM,IAAI,MAAM;QAAE,OAAO,CAAC,IAAI,CAAC,CAAC;IAEzC,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,IAAI,SAAS,GAAG,IAAI,CAAC;IAErB,OAAO,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC5B,IAAI,SAAS,CAAC,MAAM,IAAI,MAAM,EAAE,CAAC;YAC/B,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACtB,MAAM;QACR,CAAC;QAED,MAAM,OAAO,GAAG,cAAc,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QAClD,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;QACxC,SAAS,GAAG,SAAS,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IACvC,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,cAAc,CAAC,IAAY,EAAE,MAAc;IAClD,KAAK,IAAI,CAAC,GAAG,MAAM,EAAE,CAAC,GAAG,MAAM,GAAG,oBAAoB,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QACrE,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,IAAI;YAAE,OAAO,CAAC,GAAG,CAAC,CAAC;IAC7D,CAAC;IAED,KAAK,IAAI,CAAC,GAAG,MAAM,EAAE,CAAC,GAAG,MAAM,GAAG,oBAAoB,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QACrE,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,IAAI;YAAE,OAAO,CAAC,GAAG,CAAC,CAAC;IAC5D,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,SAAS,wBAAwB,CAAC,WAAmB;IACnD,MAAM,UAAU,GAAG,CAAC,CAAC,iBAAiB,CAAC,CAAC;IACxC,MAAM,MAAM,GAAG,WAAW,CAAC,UAAU,CAAC,UAAU,CAAC;QAC/C,CAAC,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,OAAO,EAAE,EAAE,GAAG,EAAE,WAAW,EAAE,EAAE;QACrD,CAAC,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,GAAG,EAAE,WAAW,EAAE,CAAC;IAC3C,OAAO,EAAE,eAAe,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC;AACzC,CAAC"}
@@ -0,0 +1,20 @@
1
+ import type { GitChangeStatus } from "../utils/constants.js";
2
+ export interface NotificationChannel {
3
+ initialize(): Promise<void>;
4
+ shutdown(): Promise<void>;
5
+ sendNotification(data: NotificationData, responseUrl?: string): Promise<void>;
6
+ }
7
+ export interface NotificationData {
8
+ projectName: string;
9
+ responseSummary: string;
10
+ durationMs: number;
11
+ gitChanges: GitChange[];
12
+ inputTokens: number;
13
+ outputTokens: number;
14
+ cacheCreationTokens: number;
15
+ cacheReadTokens: number;
16
+ }
17
+ export interface GitChange {
18
+ file: string;
19
+ status: GitChangeStatus;
20
+ }
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/channel/types.ts"],"names":[],"mappings":""}
@@ -1,17 +1,19 @@
1
1
  import * as p from "@clack/prompts";
2
2
  import { detectCliPrefix } from "../utils/install-detection.js";
3
+ import { t } from "../i18n/index.js";
3
4
  export function runHelp() {
4
5
  const prefix = detectCliPrefix();
5
- p.intro("🤖 ccbot — Claude Code ↔ Telegram Notification Bot");
6
+ p.intro(t("help.intro"));
6
7
  p.log.message([
7
- `Usage: ${prefix} [command]`,
8
+ t("help.usage", { prefix }),
8
9
  "",
9
- "Commands:",
10
- " (none) Start the bot",
11
- " setup Interactive setup (config + hooks)",
12
- " update Update ccbot to latest version",
13
- " uninstall Remove all ccbot data and hooks",
14
- " help Show this help message",
10
+ t("help.commands"),
11
+ t("help.cmdNone"),
12
+ t("help.cmdSetup"),
13
+ t("help.cmdUpdate"),
14
+ t("help.cmdUninstall"),
15
+ t("help.cmdHelp"),
15
16
  ].join("\n"));
16
- p.outro("docs → https://github.com/palooza-kaida/ccbot");
17
+ p.outro(t("help.docs"));
17
18
  }
19
+ //# sourceMappingURL=help.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"help.js","sourceRoot":"","sources":["../../src/commands/help.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,gBAAgB,CAAC;AACpC,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAC;AAChE,OAAO,EAAE,CAAC,EAAE,MAAM,kBAAkB,CAAC;AAErC,MAAM,UAAU,OAAO;IACrB,MAAM,MAAM,GAAG,eAAe,EAAE,CAAC;IAEjC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;IAEzB,CAAC,CAAC,GAAG,CAAC,OAAO,CACX;QACE,CAAC,CAAC,YAAY,EAAE,EAAE,MAAM,EAAE,CAAC;QAC3B,EAAE;QACF,CAAC,CAAC,eAAe,CAAC;QAClB,CAAC,CAAC,cAAc,CAAC;QACjB,CAAC,CAAC,eAAe,CAAC;QAClB,CAAC,CAAC,gBAAgB,CAAC;QACnB,CAAC,CAAC,mBAAmB,CAAC;QACtB,CAAC,CAAC,cAAc,CAAC;KAClB,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;IAEF,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC;AAC1B,CAAC"}
@@ -1,53 +1,70 @@
1
1
  import * as p from "@clack/prompts";
2
- import { mkdirSync, writeFileSync, readFileSync } from "node:fs";
3
- import { join } from "node:path";
4
- import { homedir } from "node:os";
5
2
  import { ConfigManager } from "../config-manager.js";
6
3
  import { HookInstaller } from "../hook/hook-installer.js";
7
4
  import { detectCliPrefix } from "../utils/install-detection.js";
8
- import { formatError } from "../utils/error-utils.js";
5
+ import { t, setLocale, SUPPORTED_LOCALES, LOCALE_LABELS } from "../i18n/index.js";
6
+ import { DEFAULT_HOOK_PORT } from "../utils/constants.js";
9
7
  export async function runSetup() {
10
- p.intro("🤖 ccbot setup");
8
+ p.intro(t("setup.intro"));
11
9
  let existing = null;
12
10
  try {
13
11
  existing = ConfigManager.load();
14
12
  }
15
- catch { }
13
+ catch {
14
+ // first-time setup, no existing config
15
+ }
16
+ const locale = await promptLanguage(existing);
17
+ setLocale(locale);
16
18
  const credentials = await promptCredentials(existing);
17
- const config = buildConfig(credentials, existing);
19
+ const config = buildConfig(credentials, existing, locale);
18
20
  saveConfig(config);
19
21
  installHook(config);
20
22
  registerChatId(config.user_id);
21
23
  const startCommand = detectCliPrefix();
22
- p.outro(`🎉 Setup complete!\n\n Next steps:\n 1. Start bot: ${startCommand}\n 2. Use Claude Code normally → notifications will arrive`);
24
+ p.outro(t("setup.complete", { command: startCommand }));
25
+ }
26
+ async function promptLanguage(existing) {
27
+ const result = await p.select({
28
+ message: t("setup.languageMessage"),
29
+ initialValue: existing?.locale ?? "en",
30
+ options: SUPPORTED_LOCALES.map((loc) => ({
31
+ value: loc,
32
+ label: LOCALE_LABELS[loc],
33
+ })),
34
+ });
35
+ if (p.isCancel(result)) {
36
+ p.cancel(t("setup.cancelled"));
37
+ process.exit(0);
38
+ }
39
+ return result;
23
40
  }
24
41
  async function promptCredentials(existing) {
25
42
  const result = await p.group({
26
43
  token: () => p.text({
27
- message: "Telegram Bot Token",
28
- placeholder: "Get from @BotFather → /newbot",
44
+ message: t("setup.tokenMessage"),
45
+ placeholder: t("setup.tokenPlaceholder"),
29
46
  initialValue: existing?.telegram_bot_token ?? "",
30
47
  validate(value) {
31
48
  if (!value || !value.trim())
32
- return "Bot token is required";
49
+ return t("setup.tokenRequired");
33
50
  if (!value.includes(":"))
34
- return "Invalid format (expected: 123456:ABC-xxx)";
51
+ return t("setup.tokenInvalidFormat");
35
52
  },
36
53
  }),
37
54
  userId: () => p.text({
38
- message: "Your Telegram User ID",
39
- placeholder: "Send /start to @userinfobot",
55
+ message: t("setup.userIdMessage"),
56
+ placeholder: t("setup.userIdPlaceholder"),
40
57
  initialValue: existing?.user_id?.toString() ?? "",
41
58
  validate(value) {
42
59
  if (!value || !value.trim())
43
- return "User ID is required";
60
+ return t("setup.userIdRequired");
44
61
  if (isNaN(parseInt(value, 10)))
45
- return "Must be a number";
62
+ return t("setup.userIdMustBeNumber");
46
63
  },
47
64
  }),
48
65
  }, {
49
66
  onCancel: () => {
50
- p.cancel("Setup cancelled.");
67
+ p.cancel(t("setup.cancelled"));
51
68
  process.exit(0);
52
69
  },
53
70
  });
@@ -56,48 +73,40 @@ async function promptCredentials(existing) {
56
73
  userId: parseInt(result.userId.trim(), 10),
57
74
  };
58
75
  }
59
- function buildConfig(credentials, existing) {
76
+ function buildConfig(credentials, existing, locale) {
60
77
  return {
61
78
  telegram_bot_token: credentials.token,
62
79
  user_id: credentials.userId,
63
- hook_port: existing?.hook_port || 9377,
80
+ hook_port: existing?.hook_port || DEFAULT_HOOK_PORT,
64
81
  hook_secret: existing?.hook_secret || ConfigManager.generateSecret(),
82
+ locale,
65
83
  };
66
84
  }
67
85
  function saveConfig(config) {
68
86
  ConfigManager.save(config);
69
- p.log.success("Config saved");
87
+ p.log.success(t("setup.configSaved"));
70
88
  }
71
89
  function installHook(config) {
90
+ if (HookInstaller.isInstalled()) {
91
+ p.log.step(t("setup.hookAlreadyInstalled"));
92
+ return;
93
+ }
72
94
  try {
73
95
  HookInstaller.install(config.hook_port, config.hook_secret);
74
- p.log.success("Hook installed → ~/.claude/settings.json");
96
+ p.log.success(t("setup.hookInstalled"));
75
97
  }
76
98
  catch (err) {
77
- const msg = formatError(err);
78
- if (msg.includes("already installed")) {
79
- p.log.step("Hook already installed");
80
- }
81
- else {
82
- p.log.error(`Hook installation failed: ${msg}`);
83
- throw new Error(`install hook: ${msg}`);
84
- }
99
+ p.log.error(t("setup.hookFailed", { error: err instanceof Error ? err.message : String(err) }));
100
+ throw err;
85
101
  }
86
102
  }
87
103
  function registerChatId(userId) {
88
- const stateDir = join(homedir(), ".ccbot");
89
- const stateFile = join(stateDir, "state.json");
90
- let state = { chat_id: null };
91
- try {
92
- const data = readFileSync(stateFile, "utf-8");
93
- state = JSON.parse(data);
94
- }
95
- catch { }
104
+ const state = ConfigManager.loadChatState();
96
105
  if (state.chat_id === userId) {
97
106
  return;
98
107
  }
99
108
  state.chat_id = userId;
100
- mkdirSync(stateDir, { recursive: true });
101
- writeFileSync(stateFile, JSON.stringify(state, null, 2), { mode: 0o600 });
102
- p.log.success("Chat ID registered");
109
+ ConfigManager.saveChatState(state);
110
+ p.log.success(t("setup.chatIdRegistered"));
103
111
  }
112
+ //# sourceMappingURL=setup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.js","sourceRoot":"","sources":["../../src/commands/setup.ts"],"names":[],"mappings":"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"}
@@ -1,41 +1,43 @@
1
1
  import * as p from "@clack/prompts";
2
2
  import { rmSync } from "node:fs";
3
- import { join } from "node:path";
4
- import { homedir } from "node:os";
5
3
  import { HookInstaller } from "../hook/hook-installer.js";
6
4
  import { detectInstallMethod } from "../utils/install-detection.js";
5
+ import { InstallMethod } from "../utils/constants.js";
6
+ import { t } from "../i18n/index.js";
7
+ import { paths } from "../utils/paths.js";
7
8
  export function runUninstall() {
8
- p.intro("🗑️ Uninstalling ccbot");
9
+ p.intro(t("uninstall.intro"));
9
10
  removeHook();
10
11
  removeConfigDirectory();
11
12
  printPostUninstallHint();
12
- p.outro("ccbot uninstalled");
13
+ p.outro(t("uninstall.done"));
13
14
  }
14
15
  function removeHook() {
15
16
  try {
16
17
  HookInstaller.uninstall();
17
- p.log.success("Hook removed from ~/.claude/settings.json");
18
+ p.log.success(t("uninstall.hookRemoved"));
18
19
  }
19
20
  catch {
20
- p.log.warn("No hook found (already removed)");
21
+ p.log.warn(t("uninstall.hookNotFound"));
21
22
  }
22
23
  }
23
24
  function removeConfigDirectory() {
24
- const ccbotDir = join(homedir(), ".ccbot");
25
+ const ccbotDir = paths.ccbotDir;
25
26
  try {
26
27
  rmSync(ccbotDir, { recursive: true, force: true });
27
- p.log.success("Removed ~/.ccbot/ (config, state, hooks)");
28
+ p.log.success(t("uninstall.configRemoved"));
28
29
  }
29
30
  catch {
30
- p.log.warn("~/.ccbot/ not found (already removed)");
31
+ p.log.warn(t("uninstall.configNotFound"));
31
32
  }
32
33
  }
33
34
  function printPostUninstallHint() {
34
35
  const method = detectInstallMethod();
35
- if (method === "global") {
36
- p.log.info("To also remove the package:\n pnpm remove -g ccbot");
36
+ if (method === InstallMethod.Global) {
37
+ p.log.info(t("uninstall.removeGlobal"));
37
38
  }
38
- else if (method === "git-clone") {
39
- p.log.info("To also remove the source:\n rm -rf <ccbot-directory>");
39
+ else if (method === InstallMethod.GitClone) {
40
+ p.log.info(t("uninstall.removeGitClone"));
40
41
  }
41
42
  }
43
+ //# sourceMappingURL=uninstall.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"uninstall.js","sourceRoot":"","sources":["../../src/commands/uninstall.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,gBAAgB,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,mBAAmB,EAAE,MAAM,+BAA+B,CAAC;AACpE,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,EAAE,CAAC,EAAE,MAAM,kBAAkB,CAAC;AACrC,OAAO,EAAE,KAAK,EAAE,MAAM,mBAAmB,CAAC;AAE1C,MAAM,UAAU,YAAY;IAC1B,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC,CAAC;IAE9B,UAAU,EAAE,CAAC;IACb,qBAAqB,EAAE,CAAC;IAExB,sBAAsB,EAAE,CAAC;IAEzB,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC;AAC/B,CAAC;AAED,SAAS,UAAU;IACjB,IAAI,CAAC;QACH,aAAa,CAAC,SAAS,EAAE,CAAC;QAC1B,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,uBAAuB,CAAC,CAAC,CAAC;IAC5C,CAAC;IAAC,MAAM,CAAC;QACP,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC;IAC1C,CAAC;AACH,CAAC;AAED,SAAS,qBAAqB;IAC5B,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;IAChC,IAAI,CAAC;QACH,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACnD,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC;IAC9C,CAAC;IAAC,MAAM,CAAC;QACP,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC;IAC5C,CAAC;AACH,CAAC;AAED,SAAS,sBAAsB;IAC7B,MAAM,MAAM,GAAG,mBAAmB,EAAE,CAAC;IAErC,IAAI,MAAM,KAAK,aAAa,CAAC,MAAM,EAAE,CAAC;QACpC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC;IAC1C,CAAC;SAAM,IAAI,MAAM,KAAK,aAAa,CAAC,QAAQ,EAAE,CAAC;QAC7C,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,0BAA0B,CAAC,CAAC,CAAC;IAC5C,CAAC;AACH,CAAC"}