ephem-cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js ADDED
@@ -0,0 +1,530 @@
1
+ #!/usr/bin/env node
2
+
3
+ // src/index.ts
4
+ import React4 from "react";
5
+ import { render } from "ink";
6
+ import { Command } from "commander";
7
+
8
+ // src/ui/App.tsx
9
+ import { useCallback, useEffect as useEffect2, useRef as useRef2, useState as useState3 } from "react";
10
+ import { Box as Box3, Text as Text3, useApp as useApp2, useInput } from "ink";
11
+
12
+ // src/ui/SetupWizard.tsx
13
+ import { useState } from "react";
14
+ import { Box, Text } from "ink";
15
+ import TextInput from "ink-text-input";
16
+ import { jsx, jsxs } from "react/jsx-runtime";
17
+ var STEPS = ["\u540E\u7AEF\u5730\u5740", "\u623F\u95F4\u7801", "\u7528\u6237\u540D"];
18
+ function SetupWizard({ defaults, onComplete }) {
19
+ const [step, setStep] = useState(0);
20
+ const [server, setServer] = useState(defaults.server ?? "");
21
+ const [room, setRoom] = useState((defaults.room ?? "").toLowerCase());
22
+ const [username, setUsername] = useState(defaults.username ?? "");
23
+ const values = [server, room, username];
24
+ const setters = [setServer, setRoom, setUsername];
25
+ function submit(value) {
26
+ const v = value.trim();
27
+ setters[step](v);
28
+ if (step === 0 && !v) {
29
+ return;
30
+ }
31
+ if (step < 2) {
32
+ setStep(step + 1);
33
+ } else {
34
+ onComplete({
35
+ server: (server || "").trim(),
36
+ room: (room || "").trim().toLowerCase(),
37
+ username: (v || "\u533F\u540D").slice(0, 32)
38
+ });
39
+ }
40
+ }
41
+ ;
42
+ return /* @__PURE__ */ jsxs(Box, { flexDirection: "column", gap: 1, children: [
43
+ /* @__PURE__ */ jsxs(Box, { flexDirection: "column", children: [
44
+ /* @__PURE__ */ jsx(Text, { color: "cyan", bold: true, children: "ephem \xB7 \u4E34\u65F6\u52A0\u5BC6\u804A\u5929\u5BA4" }),
45
+ /* @__PURE__ */ jsx(Text, { color: "gray", children: "\u6309\u56DE\u8F66\u8FDB\u5165\u4E0B\u4E00\u6B65\uFF0CCtrl+C \u9000\u51FA" })
46
+ ] }),
47
+ STEPS.map((label, i) => {
48
+ const done = i < step;
49
+ const active = i === step;
50
+ return /* @__PURE__ */ jsxs(Box, { flexDirection: "column", children: [
51
+ /* @__PURE__ */ jsxs(Text, { color: active ? "cyan" : "gray", children: [
52
+ done ? "\u2713" : active ? "?" : "\xB7",
53
+ " ",
54
+ label,
55
+ i === 0 && defaults.server ? "\uFF08\u56DE\u8F66\u4F7F\u7528\u9ED8\u8BA4\u503C\uFF09" : ""
56
+ ] }),
57
+ active ? /* @__PURE__ */ jsxs(Box, { children: [
58
+ /* @__PURE__ */ jsx(Text, { color: "gray", children: " \u203A " }),
59
+ /* @__PURE__ */ jsx(
60
+ TextInput,
61
+ {
62
+ value: values[i],
63
+ onChange: (v) => setters[i](v),
64
+ onSubmit: submit,
65
+ placeholder: i === 0 ? "wss://your-worker.workers.dev" : i === 1 ? "correct-horse-battery" : "\u4F60\u7684\u540D\u5B57"
66
+ }
67
+ )
68
+ ] }) : done ? /* @__PURE__ */ jsxs(Text, { color: "gray", children: [
69
+ " ",
70
+ values[i] || "(\u7A7A)"
71
+ ] }) : null
72
+ ] }, label);
73
+ })
74
+ ] });
75
+ }
76
+
77
+ // src/ui/ChatRoom.tsx
78
+ import { useEffect, useReducer, useRef, useState as useState2 } from "react";
79
+ import { Box as Box2, Text as Text2, useApp, useStdout } from "ink";
80
+ import TextInput2 from "ink-text-input";
81
+
82
+ // src/crypto/deriveKey.ts
83
+ import { hkdfSync } from "crypto";
84
+ var SALT = "ephem-v1-room-salt";
85
+ var INFO = "ephem-room-encryption-key";
86
+ var KEY_LEN = 32;
87
+ function deriveRoomKey(roomCode) {
88
+ const ikm = Buffer.from(roomCode, "utf8");
89
+ const salt = Buffer.from(SALT, "utf8");
90
+ const info = Buffer.from(INFO, "utf8");
91
+ return Buffer.from(hkdfSync("sha256", ikm, salt, info, KEY_LEN));
92
+ }
93
+
94
+ // src/crypto/cipher.ts
95
+ import { createCipheriv, createDecipheriv, randomBytes } from "crypto";
96
+ var ALGO = "aes-256-gcm";
97
+ var NONCE_LEN = 12;
98
+ var TAG_LEN = 16;
99
+ function encrypt(key, plaintext) {
100
+ const nonce = randomBytes(NONCE_LEN);
101
+ const cipher = createCipheriv(ALGO, key, nonce);
102
+ const enc = Buffer.concat([cipher.update(plaintext, "utf8"), cipher.final()]);
103
+ const tag = cipher.getAuthTag();
104
+ const combined = Buffer.concat([enc, tag]);
105
+ return {
106
+ ciphertext: combined.toString("base64"),
107
+ nonce: nonce.toString("base64")
108
+ };
109
+ }
110
+ function decrypt(key, payload) {
111
+ const combined = Buffer.from(payload.ciphertext, "base64");
112
+ const nonce = Buffer.from(payload.nonce, "base64");
113
+ if (combined.length < TAG_LEN + 1) {
114
+ throw new Error("\u5BC6\u6587\u957F\u5EA6\u5F02\u5E38");
115
+ }
116
+ const tag = combined.subarray(combined.length - TAG_LEN);
117
+ const enc = combined.subarray(0, combined.length - TAG_LEN);
118
+ const decipher = createDecipheriv(ALGO, key, nonce);
119
+ decipher.setAuthTag(tag);
120
+ const dec = Buffer.concat([decipher.update(enc), decipher.final()]);
121
+ return dec.toString("utf8");
122
+ }
123
+
124
+ // src/ui/ChatRoom.tsx
125
+ import { jsx as jsx2, jsxs as jsxs2 } from "react/jsx-runtime";
126
+ var lineId = 0;
127
+ function ChatRoom({ client, roomCode, username, joined, onExit }) {
128
+ const { exit } = useApp();
129
+ const { stdout } = useStdout();
130
+ const roomKey = useRef(deriveRoomKey(roomCode));
131
+ const [lines, dispatch] = useReducer(
132
+ (state, action) => {
133
+ if (action.type === "clear") return [];
134
+ return [...state, action.line].slice(-500);
135
+ },
136
+ []
137
+ );
138
+ const [input, setInput] = useState2("");
139
+ const [members, setMembers] = useState2(joined.currentMembers);
140
+ const [maxMembers] = useState2(joined.maxMembers);
141
+ const [expiresAt] = useState2(joined.expiresAt);
142
+ const [remaining, setRemaining] = useState2(() => Math.max(0, Math.floor((joined.expiresAt - Date.now()) / 1e3)));
143
+ const [closing, setClosing] = useState2(null);
144
+ const addSystem = (text) => dispatch({ type: "add", line: { id: ++lineId, kind: "system", text } });
145
+ const addMsg = (from, text, self) => dispatch({
146
+ type: "add",
147
+ line: { id: ++lineId, kind: "msg", from, text, self, time: nowStr() }
148
+ });
149
+ useEffect(() => {
150
+ addSystem(`\u5DF2\u52A0\u5165\u623F\u95F4 ${roomCode}\uFF08${joined.currentMembers}/${joined.maxMembers} \u4EBA\uFF09`);
151
+ const onPeerJoined = ({ username: u }) => {
152
+ setMembers((m) => m + 1);
153
+ addSystem(`${u} \u52A0\u5165\u4E86\u623F\u95F4`);
154
+ };
155
+ const onPeerLeft = ({ username: u }) => {
156
+ setMembers((m) => Math.max(0, m - 1));
157
+ addSystem(`${u} \u79BB\u5F00\u4E86\u623F\u95F4`);
158
+ };
159
+ const onMessage = (msg) => {
160
+ try {
161
+ const text = decrypt(roomKey.current, { ciphertext: msg.ciphertext, nonce: msg.nonce });
162
+ addMsg(msg.from, text, false);
163
+ } catch {
164
+ addSystem(`\u6536\u5230\u6765\u81EA ${msg.from} \u7684\u65E0\u6CD5\u89E3\u5BC6\u7684\u6D88\u606F`);
165
+ }
166
+ };
167
+ const onRoomClosing = ({ reason }) => {
168
+ const reasonText = reason === "ttl_expired" ? "\u623F\u95F4\u5DF2\u5230\u671F" : reason === "empty" ? "\u623F\u95F4\u5DF2\u7A7A" : "\u623F\u95F4\u88AB\u624B\u52A8\u9500\u6BC1";
169
+ setClosing(reasonText);
170
+ addSystem(`\u623F\u95F4\u5373\u5C06\u5173\u95ED\uFF1A${reasonText}`);
171
+ setTimeout(() => {
172
+ client.close();
173
+ onExit();
174
+ exit();
175
+ }, 1500);
176
+ };
177
+ const onServerError = (info) => {
178
+ addSystem(`\u9519\u8BEF\uFF1A${info.message} (${info.code})`);
179
+ };
180
+ client.on("peer_joined", onPeerJoined);
181
+ client.on("peer_left", onPeerLeft);
182
+ client.on("message", onMessage);
183
+ client.on("room_closing", onRoomClosing);
184
+ client.on("server_error", onServerError);
185
+ return () => {
186
+ client.off("peer_joined", onPeerJoined);
187
+ client.off("peer_left", onPeerLeft);
188
+ client.off("message", onMessage);
189
+ client.off("room_closing", onRoomClosing);
190
+ client.off("server_error", onServerError);
191
+ };
192
+ }, [client]);
193
+ useEffect(() => {
194
+ const t = setInterval(() => {
195
+ const r = Math.max(0, Math.floor((expiresAt - Date.now()) / 1e3));
196
+ setRemaining(r);
197
+ }, 1e3);
198
+ return () => clearInterval(t);
199
+ }, [expiresAt]);
200
+ useEffect(() => () => client.close(), [client]);
201
+ function handleSend(text) {
202
+ const t = text.trim();
203
+ if (!t) return;
204
+ try {
205
+ const { ciphertext, nonce } = encrypt(roomKey.current, t);
206
+ client.send(ciphertext, nonce);
207
+ addMsg(username, t, true);
208
+ } catch {
209
+ addSystem("\u53D1\u9001\u5931\u8D25\uFF1A\u52A0\u5BC6\u51FA\u9519");
210
+ }
211
+ setInput("");
212
+ }
213
+ const rows = stdout?.rows ?? 24;
214
+ const visible = Math.max(4, rows - 6);
215
+ const cdColor = remaining < 60 ? "red" : remaining < 300 ? "yellow" : "gray";
216
+ return /* @__PURE__ */ jsxs2(Box2, { flexDirection: "column", height: rows, children: [
217
+ /* @__PURE__ */ jsxs2(Box2, { flexDirection: "column", children: [
218
+ /* @__PURE__ */ jsxs2(Box2, { children: [
219
+ /* @__PURE__ */ jsx2(Text2, { color: "cyan", bold: true, children: "ephem" }),
220
+ /* @__PURE__ */ jsx2(Text2, { color: "gray", children: " \xB7 " }),
221
+ /* @__PURE__ */ jsx2(Text2, { bold: true, children: roomCode }),
222
+ /* @__PURE__ */ jsxs2(Text2, { color: "gray", children: [
223
+ " ",
224
+ members,
225
+ "/",
226
+ maxMembers,
227
+ " \u4EBA"
228
+ ] }),
229
+ /* @__PURE__ */ jsx2(Box2, { flexGrow: 1 }),
230
+ /* @__PURE__ */ jsxs2(Text2, { color: cdColor, children: [
231
+ "\u23F3 ",
232
+ fmtCd(remaining)
233
+ ] })
234
+ ] }),
235
+ /* @__PURE__ */ jsxs2(Text2, { color: "gray", children: [
236
+ "\u8F93\u5165\u6D88\u606F\u56DE\u8F66\u53D1\u9001 \xB7 Ctrl+C \u9000\u51FA",
237
+ closing ? ` \xB7 ${closing}` : ""
238
+ ] })
239
+ ] }),
240
+ /* @__PURE__ */ jsx2(Box2, { flexDirection: "column", flexGrow: 1, marginTop: 1, children: lines.slice(-visible).map(
241
+ (l) => l.kind === "system" ? /* @__PURE__ */ jsxs2(Text2, { color: "yellow", children: [
242
+ " ",
243
+ l.text
244
+ ] }, l.id) : /* @__PURE__ */ jsxs2(Box2, { children: [
245
+ /* @__PURE__ */ jsxs2(Text2, { color: "gray", children: [
246
+ l.time,
247
+ " "
248
+ ] }),
249
+ /* @__PURE__ */ jsx2(Text2, { color: l.self ? "cyan" : "white", bold: true, children: l.from }),
250
+ /* @__PURE__ */ jsxs2(Text2, { color: l.self ? "cyan" : "white", children: [
251
+ ": ",
252
+ l.text
253
+ ] })
254
+ ] }, l.id)
255
+ ) }),
256
+ /* @__PURE__ */ jsxs2(Box2, { marginTop: 1, children: [
257
+ /* @__PURE__ */ jsx2(Text2, { color: "cyan", children: "> " }),
258
+ /* @__PURE__ */ jsx2(
259
+ TextInput2,
260
+ {
261
+ value: input,
262
+ onChange: setInput,
263
+ onSubmit: handleSend,
264
+ placeholder: closing ? "\u623F\u95F4\u5373\u5C06\u5173\u95ED\u2026" : "\u8F93\u5165\u6D88\u606F\u2026"
265
+ }
266
+ )
267
+ ] })
268
+ ] });
269
+ }
270
+ function nowStr() {
271
+ return (/* @__PURE__ */ new Date()).toLocaleTimeString("zh-CN", { hour: "2-digit", minute: "2-digit" });
272
+ }
273
+ function fmtCd(sec) {
274
+ const h = Math.floor(sec / 3600);
275
+ const m = Math.floor(sec % 3600 / 60);
276
+ const s = sec % 60;
277
+ return `${String(h).padStart(2, "0")}:${String(m).padStart(2, "0")}:${String(s).padStart(2, "0")}`;
278
+ }
279
+
280
+ // src/ws/client.ts
281
+ import { EventEmitter } from "events";
282
+ import WebSocket from "ws";
283
+ var RoomClient = class extends EventEmitter {
284
+ constructor(server, roomCode, username) {
285
+ super();
286
+ this.server = server;
287
+ this.roomCode = roomCode;
288
+ this.username = username;
289
+ }
290
+ server;
291
+ roomCode;
292
+ username;
293
+ ws = null;
294
+ reconnectAttempt = 0;
295
+ manuallyClosed = false;
296
+ rejectedByServer = false;
297
+ pingTimer = null;
298
+ reconnectTimer = null;
299
+ connect() {
300
+ this.manuallyClosed = false;
301
+ this.rejectedByServer = false;
302
+ this.openSocket();
303
+ }
304
+ openSocket() {
305
+ const url = `${normalizeWs(this.server)}/room/${encodeURIComponent(this.roomCode)}?username=${encodeURIComponent(this.username)}`;
306
+ const ws = new WebSocket(url);
307
+ this.ws = ws;
308
+ ws.on("open", () => {
309
+ this.reconnectAttempt = 0;
310
+ this.startPing();
311
+ });
312
+ ws.on("message", (raw) => {
313
+ let msg;
314
+ try {
315
+ msg = JSON.parse(raw.toString());
316
+ } catch {
317
+ return;
318
+ }
319
+ this.dispatch(msg);
320
+ });
321
+ ws.on("unexpected-response", (_req, res) => {
322
+ let body = "";
323
+ res.on("data", (c) => body += c.toString());
324
+ res.on("end", () => {
325
+ let info = { code: `http_${res.statusCode}`, message: "\u8FDE\u63A5\u88AB\u670D\u52A1\u7AEF\u62D2\u7EDD" };
326
+ try {
327
+ const j = JSON.parse(body);
328
+ if (j.error) info = { code: String(j.error), message: String(j.message ?? j.error) };
329
+ } catch {
330
+ }
331
+ this.rejectedByServer = true;
332
+ this.emit("server_error", info);
333
+ });
334
+ });
335
+ ws.on("close", () => {
336
+ this.stopPing();
337
+ if (this.manuallyClosed || this.rejectedByServer) {
338
+ this.emit("closed");
339
+ return;
340
+ }
341
+ this.scheduleReconnect();
342
+ });
343
+ ws.on("error", () => {
344
+ });
345
+ }
346
+ dispatch(msg) {
347
+ switch (msg.type) {
348
+ case "joined":
349
+ this.emit("joined", msg.payload);
350
+ break;
351
+ case "peer_joined":
352
+ this.emit("peer_joined", msg.payload);
353
+ break;
354
+ case "peer_left":
355
+ this.emit("peer_left", msg.payload);
356
+ break;
357
+ case "message":
358
+ this.emit("message", msg.payload);
359
+ break;
360
+ case "room_closing":
361
+ this.manuallyClosed = true;
362
+ this.emit("room_closing", msg.payload);
363
+ break;
364
+ case "error":
365
+ this.emit("server_error", msg.payload);
366
+ break;
367
+ default:
368
+ break;
369
+ }
370
+ }
371
+ /** 发送一条已加密的消息(密文 + nonce)。 */
372
+ send(ciphertext, nonce) {
373
+ if (this.ws?.readyState === WebSocket.OPEN) {
374
+ this.ws.send(JSON.stringify({ type: "message", payload: { ciphertext, nonce } }));
375
+ }
376
+ }
377
+ close() {
378
+ this.manuallyClosed = true;
379
+ this.stopPing();
380
+ if (this.reconnectTimer) clearTimeout(this.reconnectTimer);
381
+ this.ws?.close();
382
+ }
383
+ startPing() {
384
+ this.stopPing();
385
+ this.pingTimer = setInterval(() => {
386
+ if (this.ws?.readyState === WebSocket.OPEN) {
387
+ this.ws.send(JSON.stringify({ type: "ping" }));
388
+ }
389
+ }, 25e3);
390
+ }
391
+ stopPing() {
392
+ if (this.pingTimer) clearInterval(this.pingTimer);
393
+ this.pingTimer = null;
394
+ }
395
+ scheduleReconnect() {
396
+ this.reconnectAttempt += 1;
397
+ const delayMs = Math.min(1e3 * 2 ** (this.reconnectAttempt - 1), 3e4);
398
+ this.emit("reconnecting", { attempt: this.reconnectAttempt, delayMs });
399
+ this.reconnectTimer = setTimeout(() => this.openSocket(), delayMs);
400
+ }
401
+ };
402
+ function normalizeWs(server) {
403
+ let s = server.trim().replace(/\/+$/, "");
404
+ if (s.startsWith("https://")) s = "wss://" + s.slice("https://".length);
405
+ else if (s.startsWith("http://")) s = "ws://" + s.slice("http://".length);
406
+ else if (!s.startsWith("ws://") && !s.startsWith("wss://")) s = "wss://" + s;
407
+ return s;
408
+ }
409
+
410
+ // src/ui/App.tsx
411
+ import { jsx as jsx3, jsxs as jsxs3 } from "react/jsx-runtime";
412
+ function App({ defaults }) {
413
+ const { exit } = useApp2();
414
+ const skipSetup = Boolean(defaults.server && defaults.room && defaults.username);
415
+ const [phase, setPhase] = useState3(skipSetup ? "connecting" : "setup");
416
+ const [client, setClient] = useState3(null);
417
+ const [joined, setJoined] = useState3(null);
418
+ const [error, setError] = useState3(null);
419
+ const cfgRef = useRef2(
420
+ skipSetup ? { server: defaults.server, room: defaults.room, username: defaults.username } : null
421
+ );
422
+ const connect = useCallback((config) => {
423
+ cfgRef.current = config;
424
+ const c = new RoomClient(config.server, config.room, config.username);
425
+ setClient(c);
426
+ setPhase("connecting");
427
+ setError(null);
428
+ setJoined(null);
429
+ c.on("joined", (info) => {
430
+ setJoined(info);
431
+ setPhase("chat");
432
+ });
433
+ c.on("server_error", (info) => {
434
+ setError(info);
435
+ setPhase("error");
436
+ });
437
+ c.on("closed", () => {
438
+ setPhase((p) => p === "connecting" ? "error" : p);
439
+ });
440
+ c.connect();
441
+ }, []);
442
+ useEffect2(() => {
443
+ if (skipSetup && cfgRef.current) connect(cfgRef.current);
444
+ }, []);
445
+ useEffect2(() => () => client?.close(), [client]);
446
+ const handleRetry = useCallback(() => {
447
+ client?.close();
448
+ setClient(null);
449
+ setJoined(null);
450
+ setError(null);
451
+ setPhase("setup");
452
+ }, [client]);
453
+ if (phase === "setup") {
454
+ return /* @__PURE__ */ jsx3(SetupWizard, { defaults, onComplete: connect });
455
+ }
456
+ if (phase === "connecting") {
457
+ return /* @__PURE__ */ jsxs3(Box3, { flexDirection: "column", gap: 1, children: [
458
+ /* @__PURE__ */ jsx3(Text3, { color: "cyan", children: "\u6B63\u5728\u8FDE\u63A5\u2026" }),
459
+ /* @__PURE__ */ jsxs3(Text3, { color: "gray", children: [
460
+ "\u670D\u52A1\u5668\uFF1A",
461
+ cfgRef.current?.server
462
+ ] }),
463
+ /* @__PURE__ */ jsxs3(Text3, { color: "gray", children: [
464
+ "\u623F\u95F4\uFF1A",
465
+ cfgRef.current?.room
466
+ ] })
467
+ ] });
468
+ }
469
+ if (phase === "error") {
470
+ return /* @__PURE__ */ jsx3(
471
+ ErrorScreen,
472
+ {
473
+ message: error?.message ?? "\u672A\u77E5\u9519\u8BEF",
474
+ code: error?.code ?? "unknown",
475
+ onRetry: handleRetry,
476
+ onExit: () => exit()
477
+ }
478
+ );
479
+ }
480
+ if (!client || !joined || !cfgRef.current) return null;
481
+ return /* @__PURE__ */ jsx3(
482
+ ChatRoom,
483
+ {
484
+ client,
485
+ roomCode: cfgRef.current.room,
486
+ username: cfgRef.current.username,
487
+ joined,
488
+ onExit: () => exit()
489
+ }
490
+ );
491
+ }
492
+ function ErrorScreen({
493
+ message,
494
+ code,
495
+ onRetry,
496
+ onExit
497
+ }) {
498
+ useInput((input, key) => {
499
+ if (key.return) onRetry();
500
+ });
501
+ return /* @__PURE__ */ jsxs3(Box3, { flexDirection: "column", gap: 1, children: [
502
+ /* @__PURE__ */ jsx3(Text3, { color: "red", bold: true, children: "\u8FDE\u63A5\u5931\u8D25" }),
503
+ /* @__PURE__ */ jsxs3(Text3, { color: "gray", children: [
504
+ message,
505
+ "\uFF08",
506
+ code,
507
+ "\uFF09"
508
+ ] }),
509
+ /* @__PURE__ */ jsx3(Text3, { color: "gray", children: "\u6309\u56DE\u8F66\u8FD4\u56DE\u8BBE\u7F6E\u91CD\u8BD5\uFF0CCtrl+C \u9000\u51FA" })
510
+ ] });
511
+ }
512
+
513
+ // src/index.ts
514
+ var program = new Command();
515
+ program.name("ephem").description("\u4E34\u65F6\u3001\u7AEF\u5230\u7AEF\u52A0\u5BC6\u7684\u547D\u4EE4\u884C\u804A\u5929\u5BA4").option("-s, --server <url>", "\u540E\u7AEF\u5730\u5740\uFF08\u4E5F\u53EF\u7528 EPHEM_SERVER \u73AF\u5883\u53D8\u91CF\uFF09").option("-r, --room <code>", "\u623F\u95F4\u7801\uFF0C\u4F8B\u5982 correct-horse-battery").option("-u, --username <name>", "\u7528\u6237\u540D").helpOption("-h, --help", "\u67E5\u770B\u5E2E\u52A9").action((opts) => {
516
+ const defaults = {
517
+ server: opts.server ?? process.env.EPHEM_SERVER,
518
+ room: opts.room,
519
+ username: opts.username
520
+ };
521
+ if (opts.room) {
522
+ process.stderr.write(
523
+ "\u26A0 \u63D0\u793A\uFF1A\u901A\u8FC7 --room \u4F20\u5165\u7684\u623F\u95F4\u7801\u53EF\u80FD\u88AB\u8BB0\u5F55\u5230 shell \u5386\u53F2\uFF0C\u5EFA\u8BAE\u4F18\u5148\u4EA4\u4E92\u5F0F\u8F93\u5165\u3002\n"
524
+ );
525
+ }
526
+ const instance = render(React4.createElement(App, { defaults }));
527
+ instance.waitUntilExit().then(() => process.exit(0)).catch(() => process.exit(1));
528
+ });
529
+ program.parse(process.argv);
530
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts","../src/ui/App.tsx","../src/ui/SetupWizard.tsx","../src/ui/ChatRoom.tsx","../src/crypto/deriveKey.ts","../src/crypto/cipher.ts","../src/ws/client.ts"],"sourcesContent":["// ephem-cli 入口:解析命令行参数,未提供的走交互式问答。\n\nimport React from \"react\";\nimport { render } from \"ink\";\nimport { Command } from \"commander\";\nimport { App } from \"./ui/App.js\";\n\nconst program = new Command();\n\nprogram\n .name(\"ephem\")\n .description(\"临时、端到端加密的命令行聊天室\")\n .option(\"-s, --server <url>\", \"后端地址(也可用 EPHEM_SERVER 环境变量)\")\n .option(\"-r, --room <code>\", \"房间码,例如 correct-horse-battery\")\n .option(\"-u, --username <name>\", \"用户名\")\n .helpOption(\"-h, --help\", \"查看帮助\")\n .action((opts) => {\n const defaults = {\n server: opts.server ?? process.env.EPHEM_SERVER,\n room: opts.room,\n username: opts.username,\n };\n\n // 安全提醒:命令行参数传房间码会被记录到 shell history,优先用交互式输入。\n if (opts.room) {\n process.stderr.write(\n \"⚠ 提示:通过 --room 传入的房间码可能被记录到 shell 历史,建议优先交互式输入。\\n\",\n );\n }\n\n const instance = render(React.createElement(App, { defaults }));\n instance.waitUntilExit()\n .then(() => process.exit(0))\n .catch(() => process.exit(1));\n });\n\nprogram.parse(process.argv);\n","import React, { useCallback, useEffect, useRef, useState } from \"react\";\nimport { Box, Text, useApp, useInput } from \"ink\";\nimport { SetupWizard, type ConnectConfig } from \"./SetupWizard.js\";\nimport { ChatRoom } from \"./ChatRoom.js\";\nimport { RoomClient, type JoinedInfo } from \"../ws/client.js\";\n\ninterface Props {\n defaults: { server?: string; room?: string; username?: string };\n}\n\ntype Phase = \"setup\" | \"connecting\" | \"chat\" | \"error\";\n\nexport function App({ defaults }: Props) {\n const { exit } = useApp();\n const skipSetup = Boolean(defaults.server && defaults.room && defaults.username);\n const [phase, setPhase] = useState<Phase>(skipSetup ? \"connecting\" : \"setup\");\n const [client, setClient] = useState<RoomClient | null>(null);\n const [joined, setJoined] = useState<JoinedInfo | null>(null);\n const [error, setError] = useState<{ code: string; message: string } | null>(null);\n const cfgRef = useRef<ConnectConfig | null>(\n skipSetup\n ? { server: defaults.server!, room: defaults.room!, username: defaults.username! }\n : null,\n );\n\n const connect = useCallback((config: ConnectConfig) => {\n cfgRef.current = config;\n const c = new RoomClient(config.server, config.room, config.username);\n setClient(c);\n setPhase(\"connecting\");\n setError(null);\n setJoined(null);\n\n c.on(\"joined\", (info: JoinedInfo) => {\n setJoined(info);\n setPhase(\"chat\");\n });\n c.on(\"server_error\", (info: { code: string; message: string }) => {\n setError(info);\n setPhase(\"error\");\n });\n // 连接彻底关闭时,若仍处于 connecting 则视为失败\n c.on(\"closed\", () => {\n setPhase((p) => (p === \"connecting\" ? \"error\" : p));\n });\n c.connect();\n }, []);\n\n // 命令行参数齐全时直接连接\n useEffect(() => {\n if (skipSetup && cfgRef.current) connect(cfgRef.current);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n // 退出清理\n useEffect(() => () => client?.close(), [client]);\n\n const handleRetry = useCallback(() => {\n client?.close();\n setClient(null);\n setJoined(null);\n setError(null);\n setPhase(\"setup\");\n }, [client]);\n\n if (phase === \"setup\") {\n return <SetupWizard defaults={defaults} onComplete={connect} />;\n }\n\n if (phase === \"connecting\") {\n return (\n <Box flexDirection=\"column\" gap={1}>\n <Text color=\"cyan\">正在连接…</Text>\n <Text color=\"gray\">服务器:{cfgRef.current?.server}</Text>\n <Text color=\"gray\">房间:{cfgRef.current?.room}</Text>\n </Box>\n );\n }\n\n if (phase === \"error\") {\n return (\n <ErrorScreen\n message={error?.message ?? \"未知错误\"}\n code={error?.code ?? \"unknown\"}\n onRetry={handleRetry}\n onExit={() => exit()}\n />\n );\n }\n\n // chat\n if (!client || !joined || !cfgRef.current) return null;\n return (\n <ChatRoom\n client={client}\n roomCode={cfgRef.current.room}\n username={cfgRef.current.username}\n joined={joined}\n onExit={() => exit()}\n />\n );\n}\n\nfunction ErrorScreen({\n message,\n code,\n onRetry,\n onExit,\n}: {\n message: string;\n code: string;\n onRetry: () => void;\n onExit: () => void;\n}) {\n useInput((input, key) => {\n if (key.return) onRetry();\n });\n return (\n <Box flexDirection=\"column\" gap={1}>\n <Text color=\"red\" bold>\n 连接失败\n </Text>\n <Text color=\"gray\">\n {message}({code})\n </Text>\n <Text color=\"gray\">按回车返回设置重试,Ctrl+C 退出</Text>\n </Box>\n );\n}\n","import React, { useState } from \"react\";\nimport { Box, Text } from \"ink\";\nimport TextInput from \"ink-text-input\";\n\nexport interface ConnectConfig {\n server: string;\n room: string;\n username: string;\n}\n\ninterface Props {\n defaults: { server?: string; room?: string; username?: string };\n onComplete: (cfg: ConnectConfig) => void;\n}\n\nconst STEPS = [\"后端地址\", \"房间码\", \"用户名\"] as const;\n\n/** 三步问答:后端地址 → 房间码 → 用户名。完成后回调 onComplete。 */\nexport function SetupWizard({ defaults, onComplete }: Props) {\n const [step, setStep] = useState(0);\n const [server, setServer] = useState(defaults.server ?? \"\");\n const [room, setRoom] = useState((defaults.room ?? \"\").toLowerCase());\n const [username, setUsername] = useState(defaults.username ?? \"\");\n\n const values = [server, room, username];\n const setters = [setServer, setRoom, setUsername];\n\n function submit(value: string) {\n const v = value.trim();\n setters[step](v);\n if (step === 0 && !v) {\n // 后端地址为空时拒绝(除非有默认值)\n return;\n }\n if (step < 2) {\n setStep(step + 1);\n } else {\n onComplete({\n server: (server || \"\").trim(),\n room: (room || \"\").trim().toLowerCase(),\n username: (v || \"匿名\").slice(0, 32),\n });\n }\n };\n\n return (\n <Box flexDirection=\"column\" gap={1}>\n <Box flexDirection=\"column\">\n <Text color=\"cyan\" bold>\n ephem · 临时加密聊天室\n </Text>\n <Text color=\"gray\">按回车进入下一步,Ctrl+C 退出</Text>\n </Box>\n\n {STEPS.map((label, i) => {\n const done = i < step;\n const active = i === step;\n return (\n <Box key={label} flexDirection=\"column\">\n <Text color={active ? \"cyan\" : \"gray\"}>\n {done ? \"✓\" : active ? \"?\" : \"·\"} {label}\n {i === 0 && defaults.server ? \"(回车使用默认值)\" : \"\"}\n </Text>\n {active ? (\n <Box>\n <Text color=\"gray\"> › </Text>\n <TextInput\n value={values[i]}\n onChange={(v) => setters[i](v)}\n onSubmit={submit}\n placeholder={i === 0 ? \"wss://your-worker.workers.dev\" : i === 1 ? \"correct-horse-battery\" : \"你的名字\"}\n />\n </Box>\n ) : done ? (\n <Text color=\"gray\"> {values[i] || \"(空)\"}</Text>\n ) : null}\n </Box>\n );\n })}\n </Box>\n );\n}\n","import React, { useEffect, useReducer, useRef, useState } from \"react\";\nimport { Box, Text, useApp, useStdout } from \"ink\";\nimport TextInput from \"ink-text-input\";\nimport type { RoomClient, JoinedInfo, ChatMessage } from \"../ws/client.js\";\nimport { deriveRoomKey } from \"../crypto/deriveKey.js\";\nimport { encrypt, decrypt } from \"../crypto/cipher.js\";\n\ninterface Props {\n client: RoomClient;\n roomCode: string;\n username: string;\n joined: JoinedInfo;\n onExit: () => void;\n}\n\ntype Line =\n | { id: number; kind: \"system\"; text: string }\n | { id: number; kind: \"msg\"; from: string; text: string; self: boolean; time: string };\n\nlet lineId = 0;\n\nexport function ChatRoom({ client, roomCode, username, joined, onExit }: Props) {\n const { exit } = useApp();\n const { stdout } = useStdout();\n const roomKey = useRef(deriveRoomKey(roomCode));\n\n const [lines, dispatch] = useReducer(\n (state: Line[], action: { type: \"add\"; line: Line } | { type: \"clear\" }) => {\n if (action.type === \"clear\") return [];\n return [...state, action.line].slice(-500);\n },\n [],\n );\n const [input, setInput] = useState(\"\");\n const [members, setMembers] = useState(joined.currentMembers);\n const [maxMembers] = useState(joined.maxMembers);\n const [expiresAt] = useState(joined.expiresAt);\n const [remaining, setRemaining] = useState(() => Math.max(0, Math.floor((joined.expiresAt - Date.now()) / 1000)));\n const [closing, setClosing] = useState<string | null>(null);\n\n const addSystem = (text: string) =>\n dispatch({ type: \"add\", line: { id: ++lineId, kind: \"system\", text } });\n const addMsg = (from: string, text: string, self: boolean) =>\n dispatch({\n type: \"add\",\n line: { id: ++lineId, kind: \"msg\", from, text, self, time: nowStr() },\n });\n\n // 订阅客户端事件\n useEffect(() => {\n addSystem(`已加入房间 ${roomCode}(${joined.currentMembers}/${joined.maxMembers} 人)`);\n\n const onPeerJoined = ({ username: u }: { username: string }) => {\n setMembers((m) => m + 1);\n addSystem(`${u} 加入了房间`);\n };\n const onPeerLeft = ({ username: u }: { username: string }) => {\n setMembers((m) => Math.max(0, m - 1));\n addSystem(`${u} 离开了房间`);\n };\n const onMessage = (msg: ChatMessage) => {\n try {\n const text = decrypt(roomKey.current, { ciphertext: msg.ciphertext, nonce: msg.nonce });\n addMsg(msg.from, text, false);\n } catch {\n addSystem(`收到来自 ${msg.from} 的无法解密的消息`);\n }\n };\n const onRoomClosing = ({ reason }: { reason: string }) => {\n const reasonText =\n reason === \"ttl_expired\" ? \"房间已到期\" : reason === \"empty\" ? \"房间已空\" : \"房间被手动销毁\";\n setClosing(reasonText);\n addSystem(`房间即将关闭:${reasonText}`);\n setTimeout(() => {\n client.close();\n onExit();\n exit();\n }, 1500);\n };\n const onServerError = (info: { code: string; message: string }) => {\n addSystem(`错误:${info.message} (${info.code})`);\n };\n\n client.on(\"peer_joined\", onPeerJoined);\n client.on(\"peer_left\", onPeerLeft);\n client.on(\"message\", onMessage);\n client.on(\"room_closing\", onRoomClosing);\n client.on(\"server_error\", onServerError);\n\n return () => {\n client.off(\"peer_joined\", onPeerJoined);\n client.off(\"peer_left\", onPeerLeft);\n client.off(\"message\", onMessage);\n client.off(\"room_closing\", onRoomClosing);\n client.off(\"server_error\", onServerError);\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [client]);\n\n // 倒计时\n useEffect(() => {\n const t = setInterval(() => {\n const r = Math.max(0, Math.floor((expiresAt - Date.now()) / 1000));\n setRemaining(r);\n }, 1000);\n return () => clearInterval(t);\n }, [expiresAt]);\n\n // 退出时关闭连接\n useEffect(() => () => client.close(), [client]);\n\n function handleSend(text: string) {\n const t = text.trim();\n if (!t) return;\n try {\n const { ciphertext, nonce } = encrypt(roomKey.current, t);\n client.send(ciphertext, nonce);\n addMsg(username, t, true);\n } catch {\n addSystem(\"发送失败:加密出错\");\n }\n setInput(\"\");\n }\n\n // 可视区域:留出 header(2) + 输入区(3) 的空间\n const rows = stdout?.rows ?? 24;\n const visible = Math.max(4, rows - 6);\n\n const cdColor = remaining < 60 ? \"red\" : remaining < 300 ? \"yellow\" : \"gray\";\n\n return (\n <Box flexDirection=\"column\" height={rows}>\n {/* Header */}\n <Box flexDirection=\"column\">\n <Box>\n <Text color=\"cyan\" bold>\n ephem\n </Text>\n <Text color=\"gray\"> · </Text>\n <Text bold>{roomCode}</Text>\n <Text color=\"gray\">\n {\" \"}\n {members}/{maxMembers} 人\n </Text>\n <Box flexGrow={1} />\n <Text color={cdColor}>⏳ {fmtCd(remaining)}</Text>\n </Box>\n <Text color=\"gray\">输入消息回车发送 · Ctrl+C 退出{closing ? ` · ${closing}` : \"\"}</Text>\n </Box>\n\n {/* 消息列表 */}\n <Box flexDirection=\"column\" flexGrow={1} marginTop={1}>\n {lines.slice(-visible).map((l) =>\n l.kind === \"system\" ? (\n <Text key={l.id} color=\"yellow\">\n {\" \"}\n {l.text}\n </Text>\n ) : (\n <Box key={l.id}>\n <Text color=\"gray\">{l.time} </Text>\n <Text color={l.self ? \"cyan\" : \"white\"} bold>\n {l.from}\n </Text>\n <Text color={l.self ? \"cyan\" : \"white\"}>: {l.text}</Text>\n </Box>\n ),\n )}\n </Box>\n\n {/* 输入栏 */}\n <Box marginTop={1}>\n <Text color=\"cyan\">{\"> \"}</Text>\n <TextInput\n value={input}\n onChange={setInput}\n onSubmit={handleSend}\n placeholder={closing ? \"房间即将关闭…\" : \"输入消息…\"}\n />\n </Box>\n </Box>\n );\n}\n\nfunction nowStr(): string {\n return new Date().toLocaleTimeString(\"zh-CN\", { hour: \"2-digit\", minute: \"2-digit\" });\n}\n\nfunction fmtCd(sec: number): string {\n const h = Math.floor(sec / 3600);\n const m = Math.floor((sec % 3600) / 60);\n const s = sec % 60;\n return `${String(h).padStart(2, \"0\")}:${String(m).padStart(2, \"0\")}:${String(s).padStart(2, \"0\")}`;\n}\n","// 从房间码派生对称密钥:HKDF(SHA-256) → 32 字节 AES-256-GCM 密钥。\n// 全程在客户端本地完成,房间码本身不因此通过网络发给后端。\n//\n// 设计说明:这是\"共享密码派生密钥\"模式(PAKE 的简化版)。房间码同时承担\n// 路由标识和密钥种子双重职责。攻击者要验证猜测必须先连上对应房间码的 WS\n// 端点,而后端对单房间码连接尝试做了限流。\n\nimport { hkdfSync } from \"node:crypto\";\n\nconst SALT = \"ephem-v1-room-salt\";\nconst INFO = \"ephem-room-encryption-key\";\nconst KEY_LEN = 32; // AES-256\n\n/** 从房间码派生房间加密密钥(32 字节)。 */\nexport function deriveRoomKey(roomCode: string): Buffer {\n const ikm = Buffer.from(roomCode, \"utf8\");\n const salt = Buffer.from(SALT, \"utf8\");\n const info = Buffer.from(INFO, \"utf8\");\n // hkdfSync 在 Node 18+ 返回 Buffer\n return Buffer.from(hkdfSync(\"sha256\", ikm, salt, info, KEY_LEN));\n}\n","// AES-256-GCM 加解密封装。\n// 约定:每条消息用独立随机 12 字节 nonce;认证标签 (authTag, 16 字节) 拼在密文末尾。\n// 密文与 nonce 都用 base64 编码传输(JSON 友好)。\n// 后端只原样转发 { ciphertext, nonce },不解密、不校验。\n\nimport { createCipheriv, createDecipheriv, randomBytes } from \"node:crypto\";\n\nconst ALGO = \"aes-256-gcm\";\nconst NONCE_LEN = 12;\nconst TAG_LEN = 16;\n\nexport interface EncryptedPayload {\n ciphertext: string; // base64(密文 + authTag)\n nonce: string; // base64(12 字节 nonce)\n}\n\n/** 加密一条文本消息。 */\nexport function encrypt(key: Buffer, plaintext: string): EncryptedPayload {\n const nonce = randomBytes(NONCE_LEN);\n const cipher = createCipheriv(ALGO, key, nonce);\n const enc = Buffer.concat([cipher.update(plaintext, \"utf8\"), cipher.final()]);\n const tag = cipher.getAuthTag();\n const combined = Buffer.concat([enc, tag]); // authTag 拼在末尾\n return {\n ciphertext: combined.toString(\"base64\"),\n nonce: nonce.toString(\"base64\"),\n };\n}\n\n/** 解密一条消息。认证失败会抛错(说明密钥不对或密文被篡改)。 */\nexport function decrypt(key: Buffer, payload: EncryptedPayload): string {\n const combined = Buffer.from(payload.ciphertext, \"base64\");\n const nonce = Buffer.from(payload.nonce, \"base64\");\n if (combined.length < TAG_LEN + 1) {\n throw new Error(\"密文长度异常\");\n }\n const tag = combined.subarray(combined.length - TAG_LEN);\n const enc = combined.subarray(0, combined.length - TAG_LEN);\n const decipher = createDecipheriv(ALGO, key, nonce);\n decipher.setAuthTag(tag);\n const dec = Buffer.concat([decipher.update(enc), decipher.final()]);\n return dec.toString(\"utf8\");\n}\n","// 房间 WebSocket 客户端:连接、收发密文、心跳保活、断线指数退避重连。\n// 服务端主动拒绝(房间不存在/已满/过期)时不重连,交由 UI 处理。\n\nimport { EventEmitter } from \"node:events\";\nimport WebSocket from \"ws\";\n\nexport interface JoinedInfo {\n username: string;\n currentMembers: number;\n maxMembers: number;\n expiresAt: number;\n}\n\nexport interface ChatMessage {\n from: string;\n ciphertext: string;\n nonce: string;\n timestamp: number;\n}\n\nexport type CloseReason = \"ttl_expired\" | \"empty\" | \"manual\";\n\ninterface ReconnectingInfo {\n attempt: number;\n delayMs: number;\n}\n\n/**\n * 事件(全部通过 on 订阅):\n * joined(info) 加入成功\n * peer_joined({username})\n * peer_left({username})\n * message(msg) 收到一条密文消息\n * room_closing({reason})房间即将销毁\n * server_error({code,message}) 服务端拒绝/出错(不可恢复)\n * reconnecting(info) 断线后准备第 N 次重连\n * closed() 连接彻底关闭\n */\nexport class RoomClient extends EventEmitter {\n private ws: WebSocket | null = null;\n private reconnectAttempt = 0;\n private manuallyClosed = false;\n private rejectedByServer = false;\n private pingTimer: NodeJS.Timeout | null = null;\n private reconnectTimer: NodeJS.Timeout | null = null;\n\n constructor(\n private readonly server: string,\n private readonly roomCode: string,\n private readonly username: string,\n ) {\n super();\n }\n\n connect(): void {\n this.manuallyClosed = false;\n this.rejectedByServer = false;\n this.openSocket();\n }\n\n private openSocket(): void {\n const url = `${normalizeWs(this.server)}/room/${encodeURIComponent(this.roomCode)}?username=${encodeURIComponent(this.username)}`;\n const ws = new WebSocket(url);\n this.ws = ws;\n\n ws.on(\"open\", () => {\n this.reconnectAttempt = 0;\n this.startPing();\n // 真正\"加入成功\"由服务端 joined 消息确认;这里只表示链路通了\n });\n\n ws.on(\"message\", (raw: Buffer | string) => {\n let msg: { type?: string; payload?: unknown };\n try {\n msg = JSON.parse(raw.toString());\n } catch {\n return;\n }\n this.dispatch(msg);\n });\n\n // 服务端返回非 101 响应(房间不存在/已满/过期/限流)\n ws.on(\"unexpected-response\", (_req, res) => {\n let body = \"\";\n res.on(\"data\", (c: Buffer) => (body += c.toString()));\n res.on(\"end\", () => {\n let info = { code: `http_${res.statusCode}`, message: \"连接被服务端拒绝\" };\n try {\n const j = JSON.parse(body);\n if (j.error) info = { code: String(j.error), message: String(j.message ?? j.error) };\n } catch {\n /* keep default */\n }\n this.rejectedByServer = true;\n this.emit(\"server_error\", info);\n });\n });\n\n ws.on(\"close\", () => {\n this.stopPing();\n if (this.manuallyClosed || this.rejectedByServer) {\n this.emit(\"closed\");\n return;\n }\n this.scheduleReconnect();\n });\n\n ws.on(\"error\", () => {\n // 网络层错误;后续 close 会触发重连流程,这里不单独抛出\n });\n }\n\n private dispatch(msg: { type?: string; payload?: any }) {\n switch (msg.type) {\n case \"joined\":\n this.emit(\"joined\", msg.payload as JoinedInfo);\n break;\n case \"peer_joined\":\n this.emit(\"peer_joined\", msg.payload);\n break;\n case \"peer_left\":\n this.emit(\"peer_left\", msg.payload);\n break;\n case \"message\":\n this.emit(\"message\", msg.payload as ChatMessage);\n break;\n case \"room_closing\":\n this.manuallyClosed = true; // 房间销毁是终态\n this.emit(\"room_closing\", msg.payload as { reason: CloseReason });\n break;\n case \"error\":\n this.emit(\"server_error\", msg.payload);\n break;\n default:\n break;\n }\n }\n\n /** 发送一条已加密的消息(密文 + nonce)。 */\n send(ciphertext: string, nonce: string): void {\n if (this.ws?.readyState === WebSocket.OPEN) {\n this.ws.send(JSON.stringify({ type: \"message\", payload: { ciphertext, nonce } }));\n }\n }\n\n close(): void {\n this.manuallyClosed = true;\n this.stopPing();\n if (this.reconnectTimer) clearTimeout(this.reconnectTimer);\n this.ws?.close();\n }\n\n private startPing(): void {\n this.stopPing();\n this.pingTimer = setInterval(() => {\n if (this.ws?.readyState === WebSocket.OPEN) {\n this.ws.send(JSON.stringify({ type: \"ping\" }));\n }\n }, 25_000);\n }\n\n private stopPing(): void {\n if (this.pingTimer) clearInterval(this.pingTimer);\n this.pingTimer = null;\n }\n\n private scheduleReconnect(): void {\n this.reconnectAttempt += 1;\n const delayMs = Math.min(1000 * 2 ** (this.reconnectAttempt - 1), 30_000);\n this.emit(\"reconnecting\", { attempt: this.reconnectAttempt, delayMs });\n this.reconnectTimer = setTimeout(() => this.openSocket(), delayMs);\n }\n}\n\n/** 把任意形式的地址规范化成 ws/wss 基础 URL(去尾部斜杠)。 */\nfunction normalizeWs(server: string): string {\n let s = server.trim().replace(/\\/+$/, \"\");\n if (s.startsWith(\"https://\")) s = \"wss://\" + s.slice(\"https://\".length);\n else if (s.startsWith(\"http://\")) s = \"ws://\" + s.slice(\"http://\".length);\n else if (!s.startsWith(\"ws://\") && !s.startsWith(\"wss://\")) s = \"wss://\" + s;\n return s;\n}\n"],"mappings":"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","names":["React","useEffect","useRef","useState","Box","Text","useApp","useState","Box","Text","TextInput","jsx","jsxs","useState","Box","Text","TextInput","jsx","jsxs","useApp","useState","useRef","useEffect","Box","Text","React"]}