@pixelbyte-software/pixcode 1.30.1 → 1.31.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 +718 -718
- package/README.de.md +248 -248
- package/README.ja.md +240 -240
- package/README.ko.md +240 -240
- package/README.md +295 -285
- package/README.ru.md +248 -248
- package/README.tr.md +250 -250
- package/README.zh-CN.md +240 -240
- package/dist/api-docs.html +879 -879
- package/dist/assets/index-BRRJ47XQ.css +32 -0
- package/dist/assets/index-EQohwyiC.js +837 -0
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/favicon.png +0 -0
- package/dist/favicon.svg +7 -8
- package/dist/generate-icons.js +48 -48
- package/dist/icons/codex-white.svg +3 -3
- package/dist/icons/codex.svg +3 -3
- package/dist/icons/cursor-white.svg +11 -11
- package/dist/icons/icon-128x128.png +0 -0
- package/dist/icons/icon-128x128.svg +9 -12
- package/dist/icons/icon-144x144.png +0 -0
- package/dist/icons/icon-144x144.svg +9 -12
- package/dist/icons/icon-152x152.png +0 -0
- package/dist/icons/icon-152x152.svg +9 -12
- package/dist/icons/icon-192x192.png +0 -0
- package/dist/icons/icon-192x192.svg +9 -12
- package/dist/icons/icon-384x384.png +0 -0
- package/dist/icons/icon-384x384.svg +9 -12
- package/dist/icons/icon-512x512.png +0 -0
- package/dist/icons/icon-512x512.svg +9 -12
- package/dist/icons/icon-72x72.png +0 -0
- package/dist/icons/icon-72x72.svg +9 -12
- package/dist/icons/icon-96x96.png +0 -0
- package/dist/icons/icon-96x96.svg +9 -12
- package/dist/icons/icon-template.svg +9 -12
- package/dist/icons/qwen-ai-icon.png +0 -0
- package/dist/index.html +59 -49
- package/dist/logo.png +0 -0
- package/dist/logo.svg +11 -16
- package/dist/manifest.json +60 -60
- package/dist/sw.js +124 -124
- package/dist-server/server/cli.js +100 -97
- package/dist-server/server/cli.js.map +1 -1
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +62 -62
- package/dist-server/server/database/db.js +114 -22
- package/dist-server/server/database/db.js.map +1 -1
- package/dist-server/server/database/schema.js +122 -89
- package/dist-server/server/database/schema.js.map +1 -1
- package/dist-server/server/gemini-cli.js +6 -1
- package/dist-server/server/gemini-cli.js.map +1 -1
- package/dist-server/server/index.js +234 -65
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +29 -2
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +22 -2
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +2 -2
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +14 -2
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js +132 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js +87 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js +201 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen.provider.js +19 -0
- package/dist-server/server/modules/providers/list/qwen/qwen.provider.js.map +1 -0
- package/dist-server/server/modules/providers/provider.registry.js +2 -0
- package/dist-server/server/modules/providers/provider.registry.js.map +1 -1
- package/dist-server/server/modules/providers/provider.routes.js +310 -1
- package/dist-server/server/modules/providers/provider.routes.js.map +1 -1
- package/dist-server/server/projects.js +197 -6
- package/dist-server/server/projects.js.map +1 -1
- package/dist-server/server/qwen-code-cli.js +350 -0
- package/dist-server/server/qwen-code-cli.js.map +1 -0
- package/dist-server/server/qwen-response-handler.js +70 -0
- package/dist-server/server/qwen-response-handler.js.map +1 -0
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +17 -17
- package/dist-server/server/routes/network.js +116 -0
- package/dist-server/server/routes/network.js.map +1 -0
- package/dist-server/server/routes/projects.js +43 -0
- package/dist-server/server/routes/projects.js.map +1 -1
- package/dist-server/server/routes/qwen.js +23 -0
- package/dist-server/server/routes/qwen.js.map +1 -0
- package/dist-server/server/routes/taskmaster.js +419 -419
- package/dist-server/server/routes/telegram.js +119 -0
- package/dist-server/server/routes/telegram.js.map +1 -0
- package/dist-server/server/services/external-access.js +228 -0
- package/dist-server/server/services/external-access.js.map +1 -0
- package/dist-server/server/services/install-jobs.js +394 -0
- package/dist-server/server/services/install-jobs.js.map +1 -0
- package/dist-server/server/services/notification-orchestrator.js +19 -5
- package/dist-server/server/services/notification-orchestrator.js.map +1 -1
- package/dist-server/server/services/provider-credentials.js +154 -0
- package/dist-server/server/services/provider-credentials.js.map +1 -0
- package/dist-server/server/services/provider-models.js +218 -0
- package/dist-server/server/services/provider-models.js.map +1 -0
- package/dist-server/server/services/telegram/bot.js +259 -0
- package/dist-server/server/services/telegram/bot.js.map +1 -0
- package/dist-server/server/services/telegram/translations.js +160 -0
- package/dist-server/server/services/telegram/translations.js.map +1 -0
- package/dist-server/server/utils/port-access.js +196 -0
- package/dist-server/server/utils/port-access.js.map +1 -0
- package/dist-server/shared/modelConstants.js +18 -0
- package/dist-server/shared/modelConstants.js.map +1 -1
- package/package.json +177 -168
- package/scripts/fix-node-pty.js +67 -67
- package/server/claude-sdk.js +834 -834
- package/server/cli.js +940 -937
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +342 -342
- package/server/daemon/manager.js +564 -564
- package/server/daemon-manager.js +920 -920
- package/server/database/db.js +696 -593
- package/server/database/schema.js +138 -102
- package/server/gemini-cli.js +475 -469
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +2730 -2557
- package/server/load-env.js +34 -34
- package/server/middleware/auth.js +132 -132
- package/server/modules/providers/list/claude/claude-auth.provider.ts +145 -123
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
- package/server/modules/providers/list/claude/claude.provider.ts +15 -15
- package/server/modules/providers/list/codex/codex-auth.provider.ts +115 -100
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
- package/server/modules/providers/list/codex/codex.provider.ts +15 -15
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -143
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +163 -151
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
- package/server/modules/providers/list/qwen/qwen-auth.provider.ts +145 -0
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -0
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +218 -0
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -0
- package/server/modules/providers/provider.registry.ts +38 -36
- package/server/modules/providers/provider.routes.ts +583 -217
- package/server/modules/providers/services/mcp.service.ts +94 -94
- package/server/modules/providers/services/provider-auth.service.ts +26 -26
- package/server/modules/providers/services/sessions.service.ts +45 -45
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
- package/server/modules/providers/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +426 -426
- package/server/projects.js +2993 -2792
- package/server/qwen-code-cli.js +392 -0
- package/server/qwen-response-handler.js +73 -0
- package/server/routes/agent.js +1245 -1245
- package/server/routes/auth.js +134 -134
- package/server/routes/codex.js +19 -19
- package/server/routes/commands.js +554 -554
- package/server/routes/cursor.js +52 -52
- package/server/routes/gemini.js +24 -24
- package/server/routes/git.js +1488 -1488
- package/server/routes/mcp-utils.js +31 -31
- package/server/routes/messages.js +61 -61
- package/server/routes/network.js +128 -0
- package/server/routes/plugins.js +307 -307
- package/server/routes/projects.js +675 -627
- package/server/routes/qwen.js +27 -0
- package/server/routes/settings.js +286 -286
- package/server/routes/taskmaster.js +1471 -1471
- package/server/routes/telegram.js +125 -0
- package/server/routes/user.js +123 -123
- package/server/services/external-access.js +240 -0
- package/server/services/install-jobs.js +410 -0
- package/server/services/notification-orchestrator.js +242 -227
- package/server/services/provider-credentials.js +151 -0
- package/server/services/provider-models.js +225 -0
- package/server/services/telegram/bot.js +280 -0
- package/server/services/telegram/translations.js +170 -0
- package/server/services/vapid-keys.js +35 -35
- package/server/sessionManager.js +225 -225
- package/server/shared/interfaces.ts +54 -54
- package/server/shared/types.ts +172 -172
- package/server/shared/utils.ts +193 -193
- package/server/tsconfig.json +36 -36
- package/server/utils/colors.js +21 -21
- package/server/utils/commandParser.js +303 -303
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/mcp-detector.js +147 -147
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +184 -184
- package/server/utils/port-access.js +209 -0
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/taskmaster-websocket.js +128 -128
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +78 -78
- package/shared/modelConstants.js +117 -97
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-C2c9QNwK.css +0 -32
- package/dist/assets/index-DyXDZED-.js +0 -1277
- package/dist-server/server/routes/cli-auth.js +0 -25
- package/dist-server/server/routes/cli-auth.js.map +0 -1
- package/server/routes/cli-auth.js +0 -27
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import express from 'express';
|
|
2
|
+
import { telegramLinksDb } from '../database/db.js';
|
|
3
|
+
import { createPairingCode, getBotState, getPublicConfig, removeBotConfig, startBot, stopBot, } from '../services/telegram/bot.js';
|
|
4
|
+
import { SUPPORTED_LANGUAGES } from '../services/telegram/translations.js';
|
|
5
|
+
const router = express.Router();
|
|
6
|
+
const sanitizeLanguage = (raw) => {
|
|
7
|
+
if (typeof raw !== 'string')
|
|
8
|
+
return 'en';
|
|
9
|
+
return SUPPORTED_LANGUAGES.includes(raw) ? raw : 'en';
|
|
10
|
+
};
|
|
11
|
+
// GET /api/telegram/status — combined bot + personal link state
|
|
12
|
+
router.get('/status', (req, res) => {
|
|
13
|
+
try {
|
|
14
|
+
const bot = getBotState();
|
|
15
|
+
const config = getPublicConfig();
|
|
16
|
+
const link = telegramLinksDb.getByUserId(req.user.id);
|
|
17
|
+
res.json({
|
|
18
|
+
bot: { ...bot, ...config },
|
|
19
|
+
link: link
|
|
20
|
+
? {
|
|
21
|
+
paired: Boolean(link.chat_id && link.verified_at),
|
|
22
|
+
telegramUsername: link.telegram_username,
|
|
23
|
+
language: link.language,
|
|
24
|
+
notificationsEnabled: Boolean(link.notifications_enabled),
|
|
25
|
+
bridgeEnabled: Boolean(link.bridge_enabled),
|
|
26
|
+
pairingCode: link.pairing_code,
|
|
27
|
+
pairingExpiresAt: link.pairing_code_expires_at,
|
|
28
|
+
verifiedAt: link.verified_at,
|
|
29
|
+
}
|
|
30
|
+
: null,
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
console.error('telegram/status failed:', error);
|
|
35
|
+
res.status(500).json({ error: 'Failed to read Telegram status' });
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
// POST /api/telegram/bot — save token and start the bot
|
|
39
|
+
router.post('/bot', async (req, res) => {
|
|
40
|
+
const { token } = req.body || {};
|
|
41
|
+
if (typeof token !== 'string' || token.length < 10) {
|
|
42
|
+
return res.status(400).json({ error: 'A valid bot token is required' });
|
|
43
|
+
}
|
|
44
|
+
try {
|
|
45
|
+
const info = await startBot({ token });
|
|
46
|
+
res.json({ success: true, bot: { ...getBotState(), configured: true, username: info.username } });
|
|
47
|
+
}
|
|
48
|
+
catch (error) {
|
|
49
|
+
console.error('telegram/bot start failed:', error);
|
|
50
|
+
const status = error?.code === 'INVALID_TOKEN' ? 400 : 502;
|
|
51
|
+
res.status(status).json({ error: error?.message || 'Failed to start bot' });
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
// DELETE /api/telegram/bot — stop and remove the configured bot
|
|
55
|
+
router.delete('/bot', async (req, res) => {
|
|
56
|
+
try {
|
|
57
|
+
await removeBotConfig();
|
|
58
|
+
res.json({ success: true, bot: getBotState() });
|
|
59
|
+
}
|
|
60
|
+
catch (error) {
|
|
61
|
+
console.error('telegram/bot remove failed:', error);
|
|
62
|
+
res.status(502).json({ error: 'Failed to remove bot' });
|
|
63
|
+
}
|
|
64
|
+
});
|
|
65
|
+
// POST /api/telegram/bot/stop — stop polling but keep the token
|
|
66
|
+
router.post('/bot/stop', async (req, res) => {
|
|
67
|
+
try {
|
|
68
|
+
await stopBot();
|
|
69
|
+
res.json({ success: true, bot: getBotState() });
|
|
70
|
+
}
|
|
71
|
+
catch (error) {
|
|
72
|
+
console.error('telegram/bot/stop failed:', error);
|
|
73
|
+
res.status(502).json({ error: 'Failed to stop bot' });
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
// POST /api/telegram/pairing-code — (re)generate a 6-digit code for this user
|
|
77
|
+
router.post('/pairing-code', (req, res) => {
|
|
78
|
+
try {
|
|
79
|
+
const language = sanitizeLanguage(req.body?.language);
|
|
80
|
+
const { code, expiresAt } = createPairingCode(req.user.id, language);
|
|
81
|
+
res.json({ success: true, code, expiresAt, language });
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
console.error('telegram/pairing-code failed:', error);
|
|
85
|
+
res.status(500).json({ error: 'Failed to generate pairing code' });
|
|
86
|
+
}
|
|
87
|
+
});
|
|
88
|
+
// PATCH /api/telegram/link — update language / toggles on the user's link
|
|
89
|
+
router.patch('/link', (req, res) => {
|
|
90
|
+
try {
|
|
91
|
+
const { language, notificationsEnabled, bridgeEnabled } = req.body || {};
|
|
92
|
+
const payload = {};
|
|
93
|
+
if (language !== undefined)
|
|
94
|
+
payload.language = sanitizeLanguage(language);
|
|
95
|
+
if (notificationsEnabled !== undefined)
|
|
96
|
+
payload.notificationsEnabled = Boolean(notificationsEnabled);
|
|
97
|
+
if (bridgeEnabled !== undefined)
|
|
98
|
+
payload.bridgeEnabled = Boolean(bridgeEnabled);
|
|
99
|
+
telegramLinksDb.updatePreferences(req.user.id, payload);
|
|
100
|
+
res.json({ success: true, link: telegramLinksDb.getByUserId(req.user.id) });
|
|
101
|
+
}
|
|
102
|
+
catch (error) {
|
|
103
|
+
console.error('telegram/link patch failed:', error);
|
|
104
|
+
res.status(500).json({ error: 'Failed to update link' });
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
// DELETE /api/telegram/link — unpair
|
|
108
|
+
router.delete('/link', (req, res) => {
|
|
109
|
+
try {
|
|
110
|
+
telegramLinksDb.unlink(req.user.id);
|
|
111
|
+
res.json({ success: true });
|
|
112
|
+
}
|
|
113
|
+
catch (error) {
|
|
114
|
+
console.error('telegram/link delete failed:', error);
|
|
115
|
+
res.status(500).json({ error: 'Failed to unpair' });
|
|
116
|
+
}
|
|
117
|
+
});
|
|
118
|
+
export default router;
|
|
119
|
+
//# sourceMappingURL=telegram.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"telegram.js","sourceRoot":"","sources":["../../../server/routes/telegram.js"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
import { spawn } from 'node:child_process';
|
|
2
|
+
import { createRequire } from 'node:module';
|
|
3
|
+
const requireCjs = createRequire(import.meta.url);
|
|
4
|
+
// nat-upnp is CommonJS and callback-based. We wrap it in promises and keep
|
|
5
|
+
// one shared client per process. The client is lazily created so importing
|
|
6
|
+
// this module does not try to bind SSDP sockets at boot.
|
|
7
|
+
let upnpClient = null;
|
|
8
|
+
const getUpnpClient = () => {
|
|
9
|
+
if (!upnpClient) {
|
|
10
|
+
const nat = requireCjs('nat-upnp');
|
|
11
|
+
upnpClient = nat.createClient();
|
|
12
|
+
}
|
|
13
|
+
return upnpClient;
|
|
14
|
+
};
|
|
15
|
+
let upnpState = {
|
|
16
|
+
mapped: false,
|
|
17
|
+
port: null,
|
|
18
|
+
externalIp: null,
|
|
19
|
+
externalUrl: null,
|
|
20
|
+
error: null,
|
|
21
|
+
};
|
|
22
|
+
// A UPnP mapping request can hang forever if the router never answers SSDP.
|
|
23
|
+
// Cap every call so the HTTP endpoint doesn't dangle — we surface a clean
|
|
24
|
+
// failure and the user can try tunnel mode instead.
|
|
25
|
+
const withTimeout = (promise, ms, label) => Promise.race([
|
|
26
|
+
promise,
|
|
27
|
+
new Promise((_, reject) => setTimeout(() => reject(new Error(`${label} timed out after ${ms}ms`)), ms)),
|
|
28
|
+
]);
|
|
29
|
+
const promisifyUpnp = (method, arg) => new Promise((resolve, reject) => {
|
|
30
|
+
const client = getUpnpClient();
|
|
31
|
+
const cb = (err, result) => (err ? reject(err) : resolve(result));
|
|
32
|
+
if (arg === undefined) {
|
|
33
|
+
client[method](cb);
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
client[method](arg, cb);
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
export const enableUpnp = async ({ port }) => {
|
|
40
|
+
upnpState = { ...upnpState, error: null };
|
|
41
|
+
try {
|
|
42
|
+
await withTimeout(promisifyUpnp('portMapping', {
|
|
43
|
+
public: port,
|
|
44
|
+
private: port,
|
|
45
|
+
// ttl:0 is documented as "never expire" — routers honor it differently,
|
|
46
|
+
// but it's the least surprising default. We leave renewal to the user
|
|
47
|
+
// clicking "enable" again if the router drops the lease.
|
|
48
|
+
ttl: 0,
|
|
49
|
+
description: 'Pixcode',
|
|
50
|
+
protocol: 'tcp',
|
|
51
|
+
}), 8000, 'UPnP portMapping');
|
|
52
|
+
const externalIp = await withTimeout(promisifyUpnp('externalIp'), 5000, 'UPnP externalIp');
|
|
53
|
+
upnpState = {
|
|
54
|
+
mapped: true,
|
|
55
|
+
port,
|
|
56
|
+
externalIp,
|
|
57
|
+
externalUrl: externalIp ? `http://${externalIp}:${port}` : null,
|
|
58
|
+
error: null,
|
|
59
|
+
};
|
|
60
|
+
return upnpState;
|
|
61
|
+
}
|
|
62
|
+
catch (err) {
|
|
63
|
+
upnpState = {
|
|
64
|
+
mapped: false,
|
|
65
|
+
port,
|
|
66
|
+
externalIp: null,
|
|
67
|
+
externalUrl: null,
|
|
68
|
+
error: err?.message || String(err),
|
|
69
|
+
};
|
|
70
|
+
throw err;
|
|
71
|
+
}
|
|
72
|
+
};
|
|
73
|
+
export const disableUpnp = async ({ port }) => {
|
|
74
|
+
try {
|
|
75
|
+
await withTimeout(promisifyUpnp('portUnmapping', { public: port, protocol: 'tcp' }), 5000, 'UPnP portUnmapping');
|
|
76
|
+
}
|
|
77
|
+
catch (err) {
|
|
78
|
+
upnpState = { ...upnpState, error: err?.message || String(err) };
|
|
79
|
+
throw err;
|
|
80
|
+
}
|
|
81
|
+
upnpState = { mapped: false, port: null, externalIp: null, externalUrl: null, error: null };
|
|
82
|
+
return upnpState;
|
|
83
|
+
};
|
|
84
|
+
export const getUpnpState = () => upnpState;
|
|
85
|
+
// ============================================================================
|
|
86
|
+
// Tunnel: detect cloudflared / ngrok and spawn; extract the public URL from
|
|
87
|
+
// stdout. We keep a single live tunnel per process — starting a new one
|
|
88
|
+
// stops the previous one to avoid dangling child processes.
|
|
89
|
+
// ============================================================================
|
|
90
|
+
let tunnelProc = null;
|
|
91
|
+
let tunnelState = {
|
|
92
|
+
running: false,
|
|
93
|
+
binary: null, // 'cloudflared' | 'ngrok'
|
|
94
|
+
url: null,
|
|
95
|
+
error: null,
|
|
96
|
+
log: [],
|
|
97
|
+
};
|
|
98
|
+
const appendLog = (line) => {
|
|
99
|
+
// Tunnels can be noisy. Cap the tail we retain so a long-running tunnel
|
|
100
|
+
// doesn't grow the log into an OOM risk.
|
|
101
|
+
tunnelState.log.push(line);
|
|
102
|
+
if (tunnelState.log.length > 200)
|
|
103
|
+
tunnelState.log.shift();
|
|
104
|
+
};
|
|
105
|
+
const detectBinary = async () => {
|
|
106
|
+
const candidates = ['cloudflared', 'ngrok'];
|
|
107
|
+
for (const name of candidates) {
|
|
108
|
+
try {
|
|
109
|
+
// `which` isn't guaranteed on Windows; we probe with `--version` instead
|
|
110
|
+
// so the same code path works on Unix and Windows Command Prompt.
|
|
111
|
+
await new Promise((resolve, reject) => {
|
|
112
|
+
const child = spawn(name, ['--version'], { stdio: 'ignore' });
|
|
113
|
+
child.on('error', reject);
|
|
114
|
+
child.on('exit', (code) => (code === 0 ? resolve() : reject(new Error(`exit ${code}`))));
|
|
115
|
+
});
|
|
116
|
+
return name;
|
|
117
|
+
}
|
|
118
|
+
catch {
|
|
119
|
+
// try next
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
return null;
|
|
123
|
+
};
|
|
124
|
+
const cloudflareUrlRegex = /https?:\/\/[a-z0-9.-]+trycloudflare\.com/i;
|
|
125
|
+
const ngrokUrlRegex = /https?:\/\/[a-z0-9.-]+\.ngrok(-free)?\.(app|io)/i;
|
|
126
|
+
const buildTunnelArgs = (binary, port) => {
|
|
127
|
+
if (binary === 'cloudflared')
|
|
128
|
+
return ['tunnel', '--url', `http://localhost:${port}`];
|
|
129
|
+
if (binary === 'ngrok')
|
|
130
|
+
return ['http', String(port), '--log', 'stdout', '--log-format', 'logfmt'];
|
|
131
|
+
throw new Error(`Unsupported tunnel binary: ${binary}`);
|
|
132
|
+
};
|
|
133
|
+
const extractUrl = (binary, text) => {
|
|
134
|
+
if (binary === 'cloudflared')
|
|
135
|
+
return text.match(cloudflareUrlRegex)?.[0] ?? null;
|
|
136
|
+
if (binary === 'ngrok')
|
|
137
|
+
return text.match(ngrokUrlRegex)?.[0] ?? null;
|
|
138
|
+
return null;
|
|
139
|
+
};
|
|
140
|
+
export const startTunnel = async ({ port }) => {
|
|
141
|
+
if (tunnelProc) {
|
|
142
|
+
// Already running — tell the caller to stop it first rather than silently
|
|
143
|
+
// replacing, which would orphan the old child and lie about state.
|
|
144
|
+
throw new Error('Tunnel already running; stop it first');
|
|
145
|
+
}
|
|
146
|
+
const binary = await detectBinary();
|
|
147
|
+
if (!binary) {
|
|
148
|
+
tunnelState = { running: false, binary: null, url: null, error: 'No tunnel binary found', log: [] };
|
|
149
|
+
const err = new Error('No tunnel binary found (tried cloudflared, ngrok)');
|
|
150
|
+
err.code = 'ENOENT_TUNNEL';
|
|
151
|
+
throw err;
|
|
152
|
+
}
|
|
153
|
+
const args = buildTunnelArgs(binary, port);
|
|
154
|
+
const child = spawn(binary, args, { stdio: ['ignore', 'pipe', 'pipe'] });
|
|
155
|
+
tunnelProc = child;
|
|
156
|
+
tunnelState = { running: true, binary, url: null, error: null, log: [] };
|
|
157
|
+
const handleChunk = (chunk) => {
|
|
158
|
+
const text = chunk.toString();
|
|
159
|
+
text.split(/\r?\n/).filter(Boolean).forEach(appendLog);
|
|
160
|
+
if (!tunnelState.url) {
|
|
161
|
+
const url = extractUrl(binary, text);
|
|
162
|
+
if (url)
|
|
163
|
+
tunnelState.url = url;
|
|
164
|
+
}
|
|
165
|
+
};
|
|
166
|
+
child.stdout.on('data', handleChunk);
|
|
167
|
+
child.stderr.on('data', handleChunk);
|
|
168
|
+
child.on('exit', (code) => {
|
|
169
|
+
tunnelProc = null;
|
|
170
|
+
tunnelState = {
|
|
171
|
+
running: false,
|
|
172
|
+
binary,
|
|
173
|
+
url: null,
|
|
174
|
+
error: code === 0 ? null : `Tunnel exited with code ${code}`,
|
|
175
|
+
log: tunnelState.log,
|
|
176
|
+
};
|
|
177
|
+
});
|
|
178
|
+
// Wait up to 15s for the public URL to appear in the log. We don't block
|
|
179
|
+
// indefinitely — if the binary is hanging on login/auth, the UI should see
|
|
180
|
+
// a clear failure instead of a spinner that never resolves.
|
|
181
|
+
const start = Date.now();
|
|
182
|
+
while (Date.now() - start < 15000) {
|
|
183
|
+
if (tunnelState.url)
|
|
184
|
+
return tunnelState;
|
|
185
|
+
if (!tunnelProc)
|
|
186
|
+
break; // process died early
|
|
187
|
+
// eslint-disable-next-line no-await-in-loop
|
|
188
|
+
await new Promise((r) => setTimeout(r, 250));
|
|
189
|
+
}
|
|
190
|
+
if (!tunnelState.url) {
|
|
191
|
+
// If we never captured a URL, kill the child so we don't leak it.
|
|
192
|
+
try {
|
|
193
|
+
child.kill();
|
|
194
|
+
}
|
|
195
|
+
catch { /* ignore */ }
|
|
196
|
+
tunnelProc = null;
|
|
197
|
+
tunnelState = { ...tunnelState, running: false, error: 'Tunnel did not report a public URL' };
|
|
198
|
+
throw new Error(tunnelState.error);
|
|
199
|
+
}
|
|
200
|
+
return tunnelState;
|
|
201
|
+
};
|
|
202
|
+
export const stopTunnel = async () => {
|
|
203
|
+
if (!tunnelProc) {
|
|
204
|
+
tunnelState = { running: false, binary: null, url: null, error: null, log: [] };
|
|
205
|
+
return tunnelState;
|
|
206
|
+
}
|
|
207
|
+
try {
|
|
208
|
+
tunnelProc.kill();
|
|
209
|
+
}
|
|
210
|
+
catch {
|
|
211
|
+
// already dead
|
|
212
|
+
}
|
|
213
|
+
tunnelProc = null;
|
|
214
|
+
tunnelState = { running: false, binary: null, url: null, error: null, log: [] };
|
|
215
|
+
return tunnelState;
|
|
216
|
+
};
|
|
217
|
+
export const getTunnelState = () => tunnelState;
|
|
218
|
+
// Explicit cleanup so the server process can shut down without leaking the
|
|
219
|
+
// child tunnel process.
|
|
220
|
+
process.on('exit', () => {
|
|
221
|
+
if (tunnelProc) {
|
|
222
|
+
try {
|
|
223
|
+
tunnelProc.kill();
|
|
224
|
+
}
|
|
225
|
+
catch { /* ignore */ }
|
|
226
|
+
}
|
|
227
|
+
});
|
|
228
|
+
//# sourceMappingURL=external-access.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"external-access.js","sourceRoot":"","sources":["../../../server/services/external-access.js"],"names":[],"mappings":"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"}
|