channelkit 1.0.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/LICENSE +21 -0
- package/README.md +829 -0
- package/config.example.yaml +37 -0
- package/dist/api/middleware/auth.d.ts +14 -0
- package/dist/api/middleware/auth.d.ts.map +1 -0
- package/dist/api/middleware/auth.js +130 -0
- package/dist/api/middleware/auth.js.map +1 -0
- package/dist/api/routes/config.d.ts +4 -0
- package/dist/api/routes/config.d.ts.map +1 -0
- package/dist/api/routes/config.js +794 -0
- package/dist/api/routes/config.js.map +1 -0
- package/dist/api/routes/dashboard.d.ts +4 -0
- package/dist/api/routes/dashboard.d.ts.map +1 -0
- package/dist/api/routes/dashboard.js +89 -0
- package/dist/api/routes/dashboard.js.map +1 -0
- package/dist/api/routes/inbound.d.ts +4 -0
- package/dist/api/routes/inbound.d.ts.map +1 -0
- package/dist/api/routes/inbound.js +293 -0
- package/dist/api/routes/inbound.js.map +1 -0
- package/dist/api/routes/logs.d.ts +4 -0
- package/dist/api/routes/logs.d.ts.map +1 -0
- package/dist/api/routes/logs.js +49 -0
- package/dist/api/routes/logs.js.map +1 -0
- package/dist/api/routes/mcp.d.ts +4 -0
- package/dist/api/routes/mcp.d.ts.map +1 -0
- package/dist/api/routes/mcp.js +100 -0
- package/dist/api/routes/mcp.js.map +1 -0
- package/dist/api/routes/restart.d.ts +4 -0
- package/dist/api/routes/restart.d.ts.map +1 -0
- package/dist/api/routes/restart.js +11 -0
- package/dist/api/routes/restart.js.map +1 -0
- package/dist/api/routes/send.d.ts +4 -0
- package/dist/api/routes/send.d.ts.map +1 -0
- package/dist/api/routes/send.js +66 -0
- package/dist/api/routes/send.js.map +1 -0
- package/dist/api/routes/settings.d.ts +4 -0
- package/dist/api/routes/settings.d.ts.map +1 -0
- package/dist/api/routes/settings.js +133 -0
- package/dist/api/routes/settings.js.map +1 -0
- package/dist/api/routes/tunnel.d.ts +4 -0
- package/dist/api/routes/tunnel.d.ts.map +1 -0
- package/dist/api/routes/tunnel.js +209 -0
- package/dist/api/routes/tunnel.js.map +1 -0
- package/dist/api/routes/twilio.d.ts +4 -0
- package/dist/api/routes/twilio.d.ts.map +1 -0
- package/dist/api/routes/twilio.js +138 -0
- package/dist/api/routes/twilio.js.map +1 -0
- package/dist/api/routes/update.d.ts +4 -0
- package/dist/api/routes/update.d.ts.map +1 -0
- package/dist/api/routes/update.js +42 -0
- package/dist/api/routes/update.js.map +1 -0
- package/dist/api/server.d.ts +52 -0
- package/dist/api/server.d.ts.map +1 -0
- package/dist/api/server.js +415 -0
- package/dist/api/server.js.map +1 -0
- package/dist/api/types.d.ts +61 -0
- package/dist/api/types.d.ts.map +1 -0
- package/dist/api/types.js +3 -0
- package/dist/api/types.js.map +1 -0
- package/dist/channels/base.d.ts +15 -0
- package/dist/channels/base.d.ts.map +1 -0
- package/dist/channels/base.js +20 -0
- package/dist/channels/base.js.map +1 -0
- package/dist/channels/email/gmail.d.ts +36 -0
- package/dist/channels/email/gmail.d.ts.map +1 -0
- package/dist/channels/email/gmail.js +351 -0
- package/dist/channels/email/gmail.js.map +1 -0
- package/dist/channels/email/index.d.ts +3 -0
- package/dist/channels/email/index.d.ts.map +1 -0
- package/dist/channels/email/index.js +8 -0
- package/dist/channels/email/index.js.map +1 -0
- package/dist/channels/email/resend.d.ts +29 -0
- package/dist/channels/email/resend.d.ts.map +1 -0
- package/dist/channels/email/resend.js +155 -0
- package/dist/channels/email/resend.js.map +1 -0
- package/dist/channels/endpoint/index.d.ts +21 -0
- package/dist/channels/endpoint/index.d.ts.map +1 -0
- package/dist/channels/endpoint/index.js +80 -0
- package/dist/channels/endpoint/index.js.map +1 -0
- package/dist/channels/sms/index.d.ts +37 -0
- package/dist/channels/sms/index.d.ts.map +1 -0
- package/dist/channels/sms/index.js +163 -0
- package/dist/channels/sms/index.js.map +1 -0
- package/dist/channels/telegram/index.d.ts +24 -0
- package/dist/channels/telegram/index.d.ts.map +1 -0
- package/dist/channels/telegram/index.js +231 -0
- package/dist/channels/telegram/index.js.map +1 -0
- package/dist/channels/voice/index.d.ts +62 -0
- package/dist/channels/voice/index.d.ts.map +1 -0
- package/dist/channels/voice/index.js +286 -0
- package/dist/channels/voice/index.js.map +1 -0
- package/dist/channels/whatsapp/index.d.ts +31 -0
- package/dist/channels/whatsapp/index.d.ts.map +1 -0
- package/dist/channels/whatsapp/index.js +383 -0
- package/dist/channels/whatsapp/index.js.map +1 -0
- package/dist/cli/commands/demo.d.ts +4 -0
- package/dist/cli/commands/demo.d.ts.map +1 -0
- package/dist/cli/commands/demo.js +55 -0
- package/dist/cli/commands/demo.js.map +1 -0
- package/dist/cli/commands/init.d.ts +2 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +254 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/install-skill.d.ts +4 -0
- package/dist/cli/commands/install-skill.d.ts.map +1 -0
- package/dist/cli/commands/install-skill.js +60 -0
- package/dist/cli/commands/install-skill.js.map +1 -0
- package/dist/cli/commands/send.d.ts +5 -0
- package/dist/cli/commands/send.d.ts.map +1 -0
- package/dist/cli/commands/send.js +46 -0
- package/dist/cli/commands/send.js.map +1 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.d.ts.map +1 -0
- package/dist/cli/commands/start.js +129 -0
- package/dist/cli/commands/start.js.map +1 -0
- package/dist/cli/helpers.d.ts +26 -0
- package/dist/cli/helpers.d.ts.map +1 -0
- package/dist/cli/helpers.js +120 -0
- package/dist/cli/helpers.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +282 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/wizards/channel.d.ts +4 -0
- package/dist/cli/wizards/channel.d.ts.map +1 -0
- package/dist/cli/wizards/channel.js +285 -0
- package/dist/cli/wizards/channel.js.map +1 -0
- package/dist/cli/wizards/provision.d.ts +4 -0
- package/dist/cli/wizards/provision.d.ts.map +1 -0
- package/dist/cli/wizards/provision.js +213 -0
- package/dist/cli/wizards/provision.js.map +1 -0
- package/dist/cli/wizards/service.d.ts +5 -0
- package/dist/cli/wizards/service.d.ts.map +1 -0
- package/dist/cli/wizards/service.js +212 -0
- package/dist/cli/wizards/service.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +6 -0
- package/dist/cli.js.map +1 -0
- package/dist/config/parser.d.ts +6 -0
- package/dist/config/parser.d.ts.map +1 -0
- package/dist/config/parser.js +37 -0
- package/dist/config/parser.js.map +1 -0
- package/dist/config/types.d.ts +170 -0
- package/dist/config/types.d.ts.map +1 -0
- package/dist/config/types.js +3 -0
- package/dist/config/types.js.map +1 -0
- package/dist/core/apiServer.d.ts +2 -0
- package/dist/core/apiServer.d.ts.map +1 -0
- package/dist/core/apiServer.js +7 -0
- package/dist/core/apiServer.js.map +1 -0
- package/dist/core/groupStore.d.ts +19 -0
- package/dist/core/groupStore.d.ts.map +1 -0
- package/dist/core/groupStore.js +48 -0
- package/dist/core/groupStore.js.map +1 -0
- package/dist/core/logger.d.ts +42 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +142 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/messageHandler.d.ts +15 -0
- package/dist/core/messageHandler.d.ts.map +1 -0
- package/dist/core/messageHandler.js +309 -0
- package/dist/core/messageHandler.js.map +1 -0
- package/dist/core/restart.d.ts +3 -0
- package/dist/core/restart.d.ts.map +1 -0
- package/dist/core/restart.js +35 -0
- package/dist/core/restart.js.map +1 -0
- package/dist/core/router.d.ts +56 -0
- package/dist/core/router.d.ts.map +1 -0
- package/dist/core/router.js +168 -0
- package/dist/core/router.js.map +1 -0
- package/dist/core/tunnel.d.ts +16 -0
- package/dist/core/tunnel.d.ts.map +1 -0
- package/dist/core/tunnel.js +99 -0
- package/dist/core/tunnel.js.map +1 -0
- package/dist/core/types.d.ts +54 -0
- package/dist/core/types.d.ts.map +1 -0
- package/dist/core/types.js +3 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/updater.d.ts +44 -0
- package/dist/core/updater.d.ts.map +1 -0
- package/dist/core/updater.js +264 -0
- package/dist/core/updater.js.map +1 -0
- package/dist/core/webhook.d.ts +26 -0
- package/dist/core/webhook.d.ts.map +1 -0
- package/dist/core/webhook.js +224 -0
- package/dist/core/webhook.js.map +1 -0
- package/dist/dashboard/assets/browser-D_-rzKir.js +8 -0
- package/dist/dashboard/assets/index-CNa084vI.js +88 -0
- package/dist/dashboard/assets/index-CRvIEyjF.css +1 -0
- package/dist/dashboard/index.html +17 -0
- package/dist/index.d.ts +22 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +551 -0
- package/dist/index.js.map +1 -0
- package/dist/mcp/index.d.ts +3 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +6 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/mcp/server.d.ts +45 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/mcp/server.js +197 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools.d.ts +16 -0
- package/dist/mcp/tools.d.ts.map +1 -0
- package/dist/mcp/tools.js +502 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/media/formatter.d.ts +12 -0
- package/dist/media/formatter.d.ts.map +1 -0
- package/dist/media/formatter.js +147 -0
- package/dist/media/formatter.js.map +1 -0
- package/dist/media/processor.d.ts +33 -0
- package/dist/media/processor.d.ts.map +1 -0
- package/dist/media/processor.js +145 -0
- package/dist/media/processor.js.map +1 -0
- package/dist/media/stt.d.ts +16 -0
- package/dist/media/stt.d.ts.map +1 -0
- package/dist/media/stt.js +298 -0
- package/dist/media/stt.js.map +1 -0
- package/dist/media/tts.d.ts +19 -0
- package/dist/media/tts.d.ts.map +1 -0
- package/dist/media/tts.js +135 -0
- package/dist/media/tts.js.map +1 -0
- package/dist/onboarding/index.d.ts +28 -0
- package/dist/onboarding/index.d.ts.map +1 -0
- package/dist/onboarding/index.js +144 -0
- package/dist/onboarding/index.js.map +1 -0
- package/dist/paths.d.ts +9 -0
- package/dist/paths.d.ts.map +1 -0
- package/dist/paths.js +14 -0
- package/dist/paths.js.map +1 -0
- package/dist/provisioning/twilio.d.ts +51 -0
- package/dist/provisioning/twilio.d.ts.map +1 -0
- package/dist/provisioning/twilio.js +175 -0
- package/dist/provisioning/twilio.js.map +1 -0
- package/echo-server.js +163 -0
- package/package.json +79 -0
|
@@ -0,0 +1,415 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.ApiServer = void 0;
|
|
40
|
+
const express_1 = __importDefault(require("express"));
|
|
41
|
+
const http_1 = require("http");
|
|
42
|
+
const helmet_1 = __importDefault(require("helmet"));
|
|
43
|
+
const express_rate_limit_1 = __importDefault(require("express-rate-limit"));
|
|
44
|
+
const ws_1 = require("ws");
|
|
45
|
+
const crypto_1 = require("crypto");
|
|
46
|
+
const auth_1 = require("./middleware/auth");
|
|
47
|
+
const send_1 = require("./routes/send");
|
|
48
|
+
const inbound_1 = require("./routes/inbound");
|
|
49
|
+
const config_1 = require("./routes/config");
|
|
50
|
+
const tunnel_1 = require("./routes/tunnel");
|
|
51
|
+
const settings_1 = require("./routes/settings");
|
|
52
|
+
const twilio_1 = require("./routes/twilio");
|
|
53
|
+
const logs_1 = require("./routes/logs");
|
|
54
|
+
const dashboard_1 = require("./routes/dashboard");
|
|
55
|
+
const restart_1 = require("./routes/restart");
|
|
56
|
+
const mcp_1 = require("./routes/mcp");
|
|
57
|
+
const update_1 = require("./routes/update");
|
|
58
|
+
/** Redact patterns that look like API keys/tokens from log text. */
|
|
59
|
+
function redactSecrets(text) {
|
|
60
|
+
// Common API key patterns (long hex, base64, bearer tokens in log output)
|
|
61
|
+
return text
|
|
62
|
+
.replace(/\b(re_[A-Za-z0-9_]{20,})\b/g, 're_****')
|
|
63
|
+
.replace(/\b(sk-[A-Za-z0-9_-]{20,})\b/g, 'sk-****')
|
|
64
|
+
.replace(/\b(whsec_[A-Za-z0-9_]{20,})\b/g, 'whsec_****')
|
|
65
|
+
.replace(/\b(AC[a-f0-9]{32})\b/g, 'AC****')
|
|
66
|
+
.replace(/\b(AIzaSy[A-Za-z0-9_-]{33})\b/g, 'AIza****')
|
|
67
|
+
.replace(/\b(eyJ[A-Za-z0-9_-]{50,})\b/g, 'eyJ****');
|
|
68
|
+
}
|
|
69
|
+
class ApiServer {
|
|
70
|
+
port;
|
|
71
|
+
app = (0, express_1.default)();
|
|
72
|
+
httpServer = (0, http_1.createServer)(this.app);
|
|
73
|
+
server = null;
|
|
74
|
+
ctx;
|
|
75
|
+
constructor(port = 4000) {
|
|
76
|
+
this.port = port;
|
|
77
|
+
const wss = new ws_1.WebSocketServer({ server: this.httpServer });
|
|
78
|
+
const broadcast = (msg) => {
|
|
79
|
+
const data = JSON.stringify(msg);
|
|
80
|
+
wss.clients.forEach((client) => {
|
|
81
|
+
if (client.readyState === ws_1.WebSocket.OPEN) {
|
|
82
|
+
client.send(data);
|
|
83
|
+
}
|
|
84
|
+
});
|
|
85
|
+
};
|
|
86
|
+
this.ctx = {
|
|
87
|
+
channels: new Map(),
|
|
88
|
+
logger: undefined,
|
|
89
|
+
configPath: undefined,
|
|
90
|
+
publicUrl: null,
|
|
91
|
+
exposeDashboard: false,
|
|
92
|
+
exposeMcp: false,
|
|
93
|
+
apiSecret: null,
|
|
94
|
+
mcpSecret: null,
|
|
95
|
+
startTime: Date.now(),
|
|
96
|
+
serverLogBuffer: [],
|
|
97
|
+
latestQR: null,
|
|
98
|
+
wss,
|
|
99
|
+
broadcast,
|
|
100
|
+
findVoiceConfig: undefined,
|
|
101
|
+
tunnelStart: undefined,
|
|
102
|
+
tunnelStop: undefined,
|
|
103
|
+
tunnelStatus: undefined,
|
|
104
|
+
mcpStart: undefined,
|
|
105
|
+
mcpStop: undefined,
|
|
106
|
+
mcpStatus: undefined,
|
|
107
|
+
updateStatus: undefined,
|
|
108
|
+
updateTrigger: undefined,
|
|
109
|
+
setPublicUrl: (url) => { this.ctx.publicUrl = url.replace(/\/$/, ''); },
|
|
110
|
+
clearPublicUrl: () => { this.ctx.publicUrl = null; },
|
|
111
|
+
getBaseUrl: () => this.ctx.publicUrl || `http://localhost:${this.port}`,
|
|
112
|
+
getReplyUrl: (channelName, jid) => `${this.ctx.getBaseUrl()}/api/send/${channelName}/${encodeURIComponent(jid)}`,
|
|
113
|
+
setExposeDashboard: (value) => { this.ctx.exposeDashboard = value; },
|
|
114
|
+
setExposeMcp: (value) => { this.ctx.exposeMcp = value; },
|
|
115
|
+
};
|
|
116
|
+
// Security headers
|
|
117
|
+
this.app.use((0, helmet_1.default)({
|
|
118
|
+
contentSecurityPolicy: {
|
|
119
|
+
directives: {
|
|
120
|
+
defaultSrc: ["'self'"],
|
|
121
|
+
scriptSrc: ["'self'", "'unsafe-inline'"],
|
|
122
|
+
styleSrc: ["'self'", "'unsafe-inline'", "https://fonts.googleapis.com"],
|
|
123
|
+
fontSrc: ["'self'", "https://fonts.gstatic.com"],
|
|
124
|
+
imgSrc: ["'self'", "data:"],
|
|
125
|
+
connectSrc: ["'self'", "ws:", "wss:"],
|
|
126
|
+
},
|
|
127
|
+
},
|
|
128
|
+
}));
|
|
129
|
+
// Body size limits
|
|
130
|
+
this.app.use(express_1.default.json({ limit: '1mb' }));
|
|
131
|
+
this.app.use((0, auth_1.mcpCors)(this.ctx));
|
|
132
|
+
this.app.use((0, auth_1.externalAccessGuard)(this.ctx));
|
|
133
|
+
this.app.use((0, auth_1.mcpAuthCheck)(this.ctx));
|
|
134
|
+
this.app.use((0, auth_1.adminAuthCheck)(this.ctx));
|
|
135
|
+
// Rate limiting — stricter for send/inbound, looser for dashboard
|
|
136
|
+
const sendLimiter = (0, express_rate_limit_1.default)({
|
|
137
|
+
windowMs: 60 * 1000,
|
|
138
|
+
max: 60,
|
|
139
|
+
standardHeaders: true,
|
|
140
|
+
legacyHeaders: false,
|
|
141
|
+
message: { error: 'Too many requests, please try again later' },
|
|
142
|
+
});
|
|
143
|
+
const inboundLimiter = (0, express_rate_limit_1.default)({
|
|
144
|
+
windowMs: 60 * 1000,
|
|
145
|
+
max: 120,
|
|
146
|
+
standardHeaders: true,
|
|
147
|
+
legacyHeaders: false,
|
|
148
|
+
message: { error: 'Too many requests' },
|
|
149
|
+
});
|
|
150
|
+
const dashboardLimiter = (0, express_rate_limit_1.default)({
|
|
151
|
+
windowMs: 60 * 1000,
|
|
152
|
+
max: 300,
|
|
153
|
+
standardHeaders: true,
|
|
154
|
+
legacyHeaders: false,
|
|
155
|
+
});
|
|
156
|
+
this.app.use('/api/send', sendLimiter);
|
|
157
|
+
this.app.use('/inbound', inboundLimiter);
|
|
158
|
+
this.app.use('/api', dashboardLimiter);
|
|
159
|
+
// Auth check endpoint — lets the dashboard verify if a secret is required/valid
|
|
160
|
+
this.app.get('/api/auth/check', (_req, res) => {
|
|
161
|
+
if (!this.ctx.apiSecret) {
|
|
162
|
+
res.json({ required: false });
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
const auth = _req.headers.authorization;
|
|
166
|
+
if (auth) {
|
|
167
|
+
const expected = `Bearer ${this.ctx.apiSecret}`;
|
|
168
|
+
const valid = auth.length === expected.length && (0, crypto_1.timingSafeEqual)(Buffer.from(auth), Buffer.from(expected));
|
|
169
|
+
res.json({ required: true, valid });
|
|
170
|
+
}
|
|
171
|
+
else {
|
|
172
|
+
res.json({ required: true, valid: false });
|
|
173
|
+
}
|
|
174
|
+
});
|
|
175
|
+
// WebSocket authentication
|
|
176
|
+
wss.on('connection', (ws, req) => {
|
|
177
|
+
if (this.ctx.apiSecret) {
|
|
178
|
+
const url = new URL(req.url || '/', `http://${req.headers.host || 'localhost'}`);
|
|
179
|
+
const token = url.searchParams.get('token');
|
|
180
|
+
if (!token || token.length !== this.ctx.apiSecret.length ||
|
|
181
|
+
!(0, crypto_1.timingSafeEqual)(Buffer.from(token), Buffer.from(this.ctx.apiSecret))) {
|
|
182
|
+
ws.close(4401, 'Unauthorized');
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
});
|
|
187
|
+
// Register all route modules
|
|
188
|
+
(0, send_1.registerSendRoutes)(this.app, this.ctx);
|
|
189
|
+
(0, inbound_1.registerInboundRoutes)(this.app, this.ctx);
|
|
190
|
+
(0, config_1.registerConfigRoutes)(this.app, this.ctx);
|
|
191
|
+
(0, tunnel_1.registerTunnelRoutes)(this.app, this.ctx);
|
|
192
|
+
(0, settings_1.registerSettingsRoutes)(this.app, this.ctx);
|
|
193
|
+
(0, twilio_1.registerTwilioRoutes)(this.app, this.ctx);
|
|
194
|
+
(0, logs_1.registerLogRoutes)(this.app, this.ctx);
|
|
195
|
+
(0, dashboard_1.registerDashboardRoutes)(this.app, this.ctx);
|
|
196
|
+
(0, restart_1.registerRestartRoutes)(this.app, this.ctx);
|
|
197
|
+
(0, mcp_1.registerMcpRoutes)(this.app, this.ctx);
|
|
198
|
+
(0, update_1.registerUpdateRoutes)(this.app, this.ctx);
|
|
199
|
+
}
|
|
200
|
+
// Proxy getters/setters that delegate to ctx for backward compat with index.ts
|
|
201
|
+
setExposeDashboard(value) { this.ctx.setExposeDashboard(value); }
|
|
202
|
+
getExposeDashboard() { return this.ctx.exposeDashboard; }
|
|
203
|
+
setApiSecret(secret) { this.ctx.apiSecret = secret != null ? String(secret) : null; }
|
|
204
|
+
setMcpSecret(secret) { this.ctx.mcpSecret = secret != null ? String(secret) : null; }
|
|
205
|
+
setLogger(logger) {
|
|
206
|
+
this.ctx.logger = logger;
|
|
207
|
+
logger.on('entry', (entry) => {
|
|
208
|
+
this.ctx.broadcast({ type: 'newEntry', entry });
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
setConfigPath(path) { this.ctx.configPath = path; }
|
|
212
|
+
captureConsole() {
|
|
213
|
+
const capture = (level, data) => {
|
|
214
|
+
const text = redactSecrets(String(data).replace(/\r?\n$/, ''));
|
|
215
|
+
if (!text)
|
|
216
|
+
return;
|
|
217
|
+
const entry = { level, text, ts: Date.now() };
|
|
218
|
+
this.ctx.serverLogBuffer.push(entry);
|
|
219
|
+
if (this.ctx.serverLogBuffer.length > 500)
|
|
220
|
+
this.ctx.serverLogBuffer.shift();
|
|
221
|
+
this.ctx.broadcast({ type: 'serverLog', ...entry });
|
|
222
|
+
};
|
|
223
|
+
const origOut = process.stdout.write.bind(process.stdout);
|
|
224
|
+
const origErr = process.stderr.write.bind(process.stderr);
|
|
225
|
+
process.stdout.write = (d, ...a) => { capture('stdout', d); return origOut(d, ...a); };
|
|
226
|
+
process.stderr.write = (d, ...a) => { capture('stderr', d); return origErr(d, ...a); };
|
|
227
|
+
}
|
|
228
|
+
registerChannel(name, channel) {
|
|
229
|
+
this.ctx.channels.set(name, channel);
|
|
230
|
+
channel.on('qr', (qr) => {
|
|
231
|
+
this.ctx.latestQR = qr;
|
|
232
|
+
this.ctx.broadcast({ type: 'qr', data: qr });
|
|
233
|
+
// Send raw QR string — frontend renders it
|
|
234
|
+
this.ctx.broadcast({ type: 'whatsapp-qr', channel: name, qr });
|
|
235
|
+
});
|
|
236
|
+
channel.on('connection', () => {
|
|
237
|
+
this.ctx.latestQR = null;
|
|
238
|
+
});
|
|
239
|
+
channel.on('connected', () => {
|
|
240
|
+
this.ctx.broadcast({ type: 'channelStatus', channel: name, connected: true, statusMessage: null });
|
|
241
|
+
this.ctx.broadcast({ type: 'whatsapp-paired', channel: name });
|
|
242
|
+
});
|
|
243
|
+
channel.on('disconnected', () => {
|
|
244
|
+
const statusMessage = channel.statusMessage || null;
|
|
245
|
+
this.ctx.broadcast({ type: 'channelStatus', channel: name, connected: false, statusMessage });
|
|
246
|
+
});
|
|
247
|
+
}
|
|
248
|
+
setPublicUrl(url) { this.ctx.setPublicUrl(url); }
|
|
249
|
+
clearPublicUrl() { this.ctx.clearPublicUrl(); }
|
|
250
|
+
getBaseUrl() { return this.ctx.getBaseUrl(); }
|
|
251
|
+
getReplyUrl(channelName, jid) { return this.ctx.getReplyUrl(channelName, jid); }
|
|
252
|
+
// Direct property setters used by index.ts — proxy to ctx
|
|
253
|
+
set findVoiceConfig(fn) { this.ctx.findVoiceConfig = fn; }
|
|
254
|
+
set tunnelStart(fn) { this.ctx.tunnelStart = fn; }
|
|
255
|
+
set tunnelStop(fn) { this.ctx.tunnelStop = fn; }
|
|
256
|
+
set tunnelStatus(fn) { this.ctx.tunnelStatus = fn; }
|
|
257
|
+
set mcpStart(fn) { this.ctx.mcpStart = fn; }
|
|
258
|
+
set mcpStop(fn) { this.ctx.mcpStop = fn; }
|
|
259
|
+
set mcpStatus(fn) { this.ctx.mcpStatus = fn; }
|
|
260
|
+
set updateStatus(fn) { this.ctx.updateStatus = fn; }
|
|
261
|
+
set updateTrigger(fn) { this.ctx.updateTrigger = fn; }
|
|
262
|
+
set reloadRouter(fn) { this.ctx.reloadRouter = fn; }
|
|
263
|
+
setExposeMcp(value) { this.ctx.setExposeMcp(value); }
|
|
264
|
+
broadcast(msg) { this.ctx.broadcast(msg); }
|
|
265
|
+
getExpressApp() { return this.app; }
|
|
266
|
+
getPort() { return this.port; }
|
|
267
|
+
async start() {
|
|
268
|
+
const portInUse = await this.isPortInUse(this.port);
|
|
269
|
+
if (portInUse) {
|
|
270
|
+
const result = await this.promptPortConflict(this.port);
|
|
271
|
+
if (result === false) {
|
|
272
|
+
console.log('[api] Exiting — port is in use.');
|
|
273
|
+
process.exit(1);
|
|
274
|
+
}
|
|
275
|
+
if (typeof result === 'number') {
|
|
276
|
+
// User chose a different port
|
|
277
|
+
this.port = result;
|
|
278
|
+
}
|
|
279
|
+
this.httpServer = (0, http_1.createServer)(this.app);
|
|
280
|
+
this.ctx.wss = new ws_1.WebSocketServer({ server: this.httpServer });
|
|
281
|
+
}
|
|
282
|
+
if (!this.ctx.apiSecret) {
|
|
283
|
+
console.warn('⚠️ No api_secret configured — dashboard and API endpoints are unprotected. Set api_secret in config.yaml for production use.');
|
|
284
|
+
}
|
|
285
|
+
return new Promise((resolve, reject) => {
|
|
286
|
+
this.server = this.httpServer.listen(this.port, () => {
|
|
287
|
+
console.log(`[api] Async API listening on port ${this.port}`);
|
|
288
|
+
console.log(`📊 Dashboard: http://localhost:${this.port}/dashboard`);
|
|
289
|
+
resolve();
|
|
290
|
+
});
|
|
291
|
+
this.server.on('error', (err) => reject(err));
|
|
292
|
+
});
|
|
293
|
+
}
|
|
294
|
+
isPortInUse(port) {
|
|
295
|
+
return new Promise((resolve) => {
|
|
296
|
+
const tester = require('net').createServer()
|
|
297
|
+
.once('error', (err) => {
|
|
298
|
+
resolve(err.code === 'EADDRINUSE');
|
|
299
|
+
})
|
|
300
|
+
.once('listening', () => {
|
|
301
|
+
tester.close(() => resolve(false));
|
|
302
|
+
})
|
|
303
|
+
.listen(port);
|
|
304
|
+
});
|
|
305
|
+
}
|
|
306
|
+
/** Returns true if process was killed, a number for an alternative port, or false to abort. */
|
|
307
|
+
async promptPortConflict(port) {
|
|
308
|
+
const { execSync } = await Promise.resolve().then(() => __importStar(require('child_process')));
|
|
309
|
+
// Resolve lsof path — macOS keeps it in /usr/sbin which may not be in PATH for background processes
|
|
310
|
+
let lsof = 'lsof';
|
|
311
|
+
try {
|
|
312
|
+
execSync('which lsof', { encoding: 'utf-8', stdio: 'pipe' });
|
|
313
|
+
}
|
|
314
|
+
catch {
|
|
315
|
+
try {
|
|
316
|
+
execSync('/usr/sbin/lsof -v', { encoding: 'utf-8', stdio: 'pipe' });
|
|
317
|
+
lsof = '/usr/sbin/lsof';
|
|
318
|
+
}
|
|
319
|
+
catch { }
|
|
320
|
+
}
|
|
321
|
+
let pid;
|
|
322
|
+
try {
|
|
323
|
+
const out = execSync(`${lsof} -ti tcp:${port}`, { encoding: 'utf-8' }).trim();
|
|
324
|
+
pid = out.split('\n')[0];
|
|
325
|
+
}
|
|
326
|
+
catch { }
|
|
327
|
+
const pidInfo = pid ? ` (PID ${pid})` : '';
|
|
328
|
+
// Detect environments where stdin is unavailable or intercepted:
|
|
329
|
+
// - Not a TTY (piped, CI, etc.)
|
|
330
|
+
// - tsx watch / node --watch: inherits TTY handles so isTTY is true,
|
|
331
|
+
// but stdin is consumed by the watch runner — readline never receives input.
|
|
332
|
+
const isWatchMode = process.execArgv.some(a => a.includes('--watch'))
|
|
333
|
+
|| process.argv.some(a => a === '--watch')
|
|
334
|
+
|| !!process.env.TSX_WATCH;
|
|
335
|
+
const autoKill = !process.stdin.isTTY || isWatchMode;
|
|
336
|
+
if (!autoKill) {
|
|
337
|
+
console.error(`\n⚠️ Port ${port} is already in use${pidInfo}.`);
|
|
338
|
+
const { createInterface } = await Promise.resolve().then(() => __importStar(require('readline')));
|
|
339
|
+
const rl = createInterface({ input: process.stdin, output: process.stderr });
|
|
340
|
+
const answer = await new Promise((res) => {
|
|
341
|
+
// Timeout: if readline is broken (e.g. stdin intercepted), auto-kill
|
|
342
|
+
const timeout = setTimeout(() => {
|
|
343
|
+
rl.close();
|
|
344
|
+
console.error(` No response received — auto-killing process on port ${port}...`);
|
|
345
|
+
res('k');
|
|
346
|
+
}, 10000);
|
|
347
|
+
rl.question(` [K]ill the existing process, [C]hange port, or [Q]uit? (auto-kill in 10s) `, (ans) => {
|
|
348
|
+
clearTimeout(timeout);
|
|
349
|
+
rl.close();
|
|
350
|
+
res(ans.trim().toLowerCase());
|
|
351
|
+
});
|
|
352
|
+
});
|
|
353
|
+
if (answer === 'c' || answer === 'change') {
|
|
354
|
+
const rl2 = createInterface({ input: process.stdin, output: process.stderr });
|
|
355
|
+
const newPortStr = await new Promise((res) => {
|
|
356
|
+
rl2.question(` Enter new port: `, (ans) => {
|
|
357
|
+
rl2.close();
|
|
358
|
+
res(ans.trim());
|
|
359
|
+
});
|
|
360
|
+
});
|
|
361
|
+
const newPort = parseInt(newPortStr, 10);
|
|
362
|
+
if (!newPort || newPort < 1 || newPort > 65535) {
|
|
363
|
+
console.error(` Invalid port number.`);
|
|
364
|
+
return false;
|
|
365
|
+
}
|
|
366
|
+
if (await this.isPortInUse(newPort)) {
|
|
367
|
+
console.error(` Port ${newPort} is also in use.`);
|
|
368
|
+
return false;
|
|
369
|
+
}
|
|
370
|
+
console.log(` Switching to port ${newPort}.`);
|
|
371
|
+
return newPort;
|
|
372
|
+
}
|
|
373
|
+
if (answer !== 'k' && answer !== 'kill' && answer !== 'y' && answer !== 'yes')
|
|
374
|
+
return false;
|
|
375
|
+
}
|
|
376
|
+
else {
|
|
377
|
+
console.error(`\n⚠️ Port ${port} is already in use${pidInfo}. Auto-killing...`);
|
|
378
|
+
}
|
|
379
|
+
try {
|
|
380
|
+
try {
|
|
381
|
+
execSync(`${lsof} -ti tcp:${port} | xargs kill -9 2>/dev/null`, { encoding: 'utf-8' });
|
|
382
|
+
}
|
|
383
|
+
catch { }
|
|
384
|
+
console.log(` Killed process on port ${port}. Waiting for port to be released...`);
|
|
385
|
+
for (let i = 0; i < 20; i++) {
|
|
386
|
+
await new Promise((r) => setTimeout(r, 500));
|
|
387
|
+
if (!(await this.isPortInUse(port)))
|
|
388
|
+
return true;
|
|
389
|
+
try {
|
|
390
|
+
execSync(`${lsof} -ti tcp:${port} | xargs kill -9 2>/dev/null`, { encoding: 'utf-8' });
|
|
391
|
+
}
|
|
392
|
+
catch { }
|
|
393
|
+
}
|
|
394
|
+
console.error(` Port ${port} is still in use after waiting.`);
|
|
395
|
+
return false;
|
|
396
|
+
}
|
|
397
|
+
catch (killErr) {
|
|
398
|
+
console.error(` Failed to kill process: ${killErr.message}`);
|
|
399
|
+
return false;
|
|
400
|
+
}
|
|
401
|
+
}
|
|
402
|
+
async stop() {
|
|
403
|
+
return new Promise((resolve) => {
|
|
404
|
+
this.ctx.wss.close();
|
|
405
|
+
if (this.server) {
|
|
406
|
+
this.server.close(() => resolve());
|
|
407
|
+
}
|
|
408
|
+
else {
|
|
409
|
+
resolve();
|
|
410
|
+
}
|
|
411
|
+
});
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
exports.ApiServer = ApiServer;
|
|
415
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/api/server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { WebSocketServer } from 'ws';
|
|
2
|
+
import { Channel } from '../channels/base';
|
|
3
|
+
import { Logger } from '../core/logger';
|
|
4
|
+
export interface ServerContext {
|
|
5
|
+
channels: Map<string, Channel>;
|
|
6
|
+
logger?: Logger;
|
|
7
|
+
configPath?: string;
|
|
8
|
+
publicUrl: string | null;
|
|
9
|
+
exposeDashboard: boolean;
|
|
10
|
+
exposeMcp: boolean;
|
|
11
|
+
apiSecret: string | null;
|
|
12
|
+
mcpSecret: string | null;
|
|
13
|
+
startTime: number;
|
|
14
|
+
serverLogBuffer: Array<{
|
|
15
|
+
level: string;
|
|
16
|
+
text: string;
|
|
17
|
+
ts: number;
|
|
18
|
+
}>;
|
|
19
|
+
latestQR: string | null;
|
|
20
|
+
wss: WebSocketServer;
|
|
21
|
+
broadcast: (msg: any) => void;
|
|
22
|
+
findVoiceConfig?: (channelName: string) => any;
|
|
23
|
+
tunnelStart?: () => Promise<{
|
|
24
|
+
url: string;
|
|
25
|
+
}>;
|
|
26
|
+
tunnelStop?: () => Promise<void>;
|
|
27
|
+
tunnelStatus?: () => {
|
|
28
|
+
active: boolean;
|
|
29
|
+
url: string | null;
|
|
30
|
+
};
|
|
31
|
+
mcpStart?: () => Promise<{
|
|
32
|
+
url: string;
|
|
33
|
+
}>;
|
|
34
|
+
mcpStop?: () => Promise<void>;
|
|
35
|
+
mcpStatus?: () => {
|
|
36
|
+
active: boolean;
|
|
37
|
+
url: string | null;
|
|
38
|
+
};
|
|
39
|
+
updateStatus?: () => Promise<{
|
|
40
|
+
mode: string;
|
|
41
|
+
currentVersion: string;
|
|
42
|
+
latestVersion: string;
|
|
43
|
+
updateAvailable: boolean;
|
|
44
|
+
behindCount: number;
|
|
45
|
+
lastChecked: number;
|
|
46
|
+
}>;
|
|
47
|
+
updateTrigger?: () => Promise<{
|
|
48
|
+
success: boolean;
|
|
49
|
+
previousVersion: string;
|
|
50
|
+
newVersion: string;
|
|
51
|
+
error?: string;
|
|
52
|
+
}>;
|
|
53
|
+
reloadRouter?: () => void;
|
|
54
|
+
setPublicUrl: (url: string) => void;
|
|
55
|
+
clearPublicUrl: () => void;
|
|
56
|
+
getBaseUrl: () => string;
|
|
57
|
+
getReplyUrl: (channelName: string, jid: string) => string;
|
|
58
|
+
setExposeDashboard: (value: boolean) => void;
|
|
59
|
+
setExposeMcp: (value: boolean) => void;
|
|
60
|
+
}
|
|
61
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/api/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAa,eAAe,EAAE,MAAM,IAAI,CAAC;AAChD,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAExC,MAAM,WAAW,aAAa;IAC5B,QAAQ,EAAE,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,eAAe,EAAE,OAAO,CAAC;IACzB,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,SAAS,EAAE,MAAM,CAAC;IAClB,eAAe,EAAE,KAAK,CAAC;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,EAAE,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACpE,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,GAAG,EAAE,eAAe,CAAC;IACrB,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,KAAK,IAAI,CAAC;IAC9B,eAAe,CAAC,EAAE,CAAC,WAAW,EAAE,MAAM,KAAK,GAAG,CAAC;IAC/C,WAAW,CAAC,EAAE,MAAM,OAAO,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAC7C,UAAU,CAAC,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACjC,YAAY,CAAC,EAAE,MAAM;QAAE,MAAM,EAAE,OAAO,CAAC;QAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAAA;KAAE,CAAC;IAC7D,QAAQ,CAAC,EAAE,MAAM,OAAO,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAC1C,OAAO,CAAC,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC9B,SAAS,CAAC,EAAE,MAAM;QAAE,MAAM,EAAE,OAAO,CAAC;QAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAAA;KAAE,CAAC;IAC1D,YAAY,CAAC,EAAE,MAAM,OAAO,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,cAAc,EAAE,MAAM,CAAC;QAAC,aAAa,EAAE,MAAM,CAAC;QAAC,eAAe,EAAE,OAAO,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAClK,aAAa,CAAC,EAAE,MAAM,OAAO,CAAC;QAAE,OAAO,EAAE,OAAO,CAAC;QAAC,eAAe,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACjH,YAAY,CAAC,EAAE,MAAM,IAAI,CAAC;IAC1B,YAAY,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IACpC,cAAc,EAAE,MAAM,IAAI,CAAC;IAC3B,UAAU,EAAE,MAAM,MAAM,CAAC;IACzB,WAAW,EAAE,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,MAAM,CAAC;IAC1D,kBAAkB,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;IAC7C,YAAY,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;CACxC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/api/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { EventEmitter } from 'events';
|
|
2
|
+
import { ChannelConfig } from '../config/types';
|
|
3
|
+
import { UnifiedMessage, WebhookResponse } from '../core/types';
|
|
4
|
+
export declare abstract class Channel extends EventEmitter {
|
|
5
|
+
readonly name: string;
|
|
6
|
+
protected config: ChannelConfig;
|
|
7
|
+
/** Whether this channel is currently connected and able to send messages. */
|
|
8
|
+
connected: boolean;
|
|
9
|
+
constructor(name: string, config: ChannelConfig);
|
|
10
|
+
abstract connect(): Promise<void>;
|
|
11
|
+
abstract disconnect(): Promise<void>;
|
|
12
|
+
abstract send(to: string, response: WebhookResponse): Promise<void>;
|
|
13
|
+
protected emitMessage(message: UnifiedMessage): void;
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=base.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.d.ts","sourceRoot":"","sources":["../../src/channels/base.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAEhE,8BAAsB,OAAQ,SAAQ,YAAY;aAK9B,IAAI,EAAE,MAAM;IAC5B,SAAS,CAAC,MAAM,EAAE,aAAa;IALjC,6EAA6E;IACtE,SAAS,UAAQ;gBAGN,IAAI,EAAE,MAAM,EAClB,MAAM,EAAE,aAAa;IAKjC,QAAQ,CAAC,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IACjC,QAAQ,CAAC,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IACpC,QAAQ,CAAC,IAAI,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAEnE,SAAS,CAAC,WAAW,CAAC,OAAO,EAAE,cAAc;CAG9C"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Channel = void 0;
|
|
4
|
+
const events_1 = require("events");
|
|
5
|
+
class Channel extends events_1.EventEmitter {
|
|
6
|
+
name;
|
|
7
|
+
config;
|
|
8
|
+
/** Whether this channel is currently connected and able to send messages. */
|
|
9
|
+
connected = true;
|
|
10
|
+
constructor(name, config) {
|
|
11
|
+
super();
|
|
12
|
+
this.name = name;
|
|
13
|
+
this.config = config;
|
|
14
|
+
}
|
|
15
|
+
emitMessage(message) {
|
|
16
|
+
this.emit('message', message);
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
exports.Channel = Channel;
|
|
20
|
+
//# sourceMappingURL=base.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base.js","sourceRoot":"","sources":["../../src/channels/base.ts"],"names":[],"mappings":";;;AAAA,mCAAsC;AAItC,MAAsB,OAAQ,SAAQ,qBAAY;IAK9B;IACN;IALZ,6EAA6E;IACtE,SAAS,GAAG,IAAI,CAAC;IAExB,YACkB,IAAY,EAClB,MAAqB;QAE/B,KAAK,EAAE,CAAC;QAHQ,SAAI,GAAJ,IAAI,CAAQ;QAClB,WAAM,GAAN,MAAM,CAAe;IAGjC,CAAC;IAMS,WAAW,CAAC,OAAuB;QAC3C,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAChC,CAAC;CACF;AAlBD,0BAkBC"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Gmail Channel — OAuth2 + polling
|
|
3
|
+
*
|
|
4
|
+
* Flow:
|
|
5
|
+
* 1. First run: opens browser for OAuth consent → saves refresh token
|
|
6
|
+
* 2. Polls for new messages at configured interval
|
|
7
|
+
* 3. Sends replies via Gmail API
|
|
8
|
+
*/
|
|
9
|
+
import { Channel } from '../base';
|
|
10
|
+
import { GmailChannelConfig } from '../../config/types';
|
|
11
|
+
import { WebhookResponse } from '../../core/types';
|
|
12
|
+
export declare class GmailChannel extends Channel {
|
|
13
|
+
private tokens;
|
|
14
|
+
private tokenPath;
|
|
15
|
+
private pollTimer;
|
|
16
|
+
private lastHistoryId;
|
|
17
|
+
private processedIds;
|
|
18
|
+
constructor(name: string, config: GmailChannelConfig);
|
|
19
|
+
private get cfg();
|
|
20
|
+
connect(): Promise<void>;
|
|
21
|
+
disconnect(): Promise<void>;
|
|
22
|
+
send(to: string, response: WebhookResponse): Promise<void>;
|
|
23
|
+
/**
|
|
24
|
+
* Send a reply to a specific thread
|
|
25
|
+
*/
|
|
26
|
+
sendReply(to: string, response: WebhookResponse, threadId: string, inReplyTo?: string): Promise<void>;
|
|
27
|
+
private startOAuthFlow;
|
|
28
|
+
private refreshAccessToken;
|
|
29
|
+
private ensureValidToken;
|
|
30
|
+
private saveTokens;
|
|
31
|
+
private gmailApi;
|
|
32
|
+
private poll;
|
|
33
|
+
private toUnified;
|
|
34
|
+
private extractBody;
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=gmail.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"gmail.d.ts","sourceRoot":"","sources":["../../../src/channels/email/gmail.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAkB,eAAe,EAAE,MAAM,kBAAkB,CAAC;AAWnE,qBAAa,YAAa,SAAQ,OAAO;IACvC,OAAO,CAAC,MAAM,CAA4B;IAC1C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,SAAS,CAA+C;IAChE,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,YAAY,CAA0B;gBAElC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,kBAAkB;IAKpD,OAAO,KAAK,GAAG,GAEd;IAEK,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IA2BxB,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAO3B,IAAI,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAqBhE;;OAEG;IACG,SAAS,CAAC,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,eAAe,EAAE,QAAQ,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;YAuB7F,cAAc;YA2Fd,kBAAkB;YAyBlB,gBAAgB;IAO9B,OAAO,CAAC,UAAU;YAQJ,QAAQ;YAsBR,IAAI;IA2ClB,OAAO,CAAC,SAAS;IA0CjB,OAAO,CAAC,WAAW;CAgBpB"}
|