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 +530 -0
- package/dist/index.js.map +1 -0
- package/integration-test.mjs +226 -0
- package/package.json +32 -0
- package/src/crypto/cipher.ts +43 -0
- package/src/crypto/deriveKey.ts +21 -0
- package/src/index.ts +37 -0
- package/src/ui/App.tsx +129 -0
- package/src/ui/ChatRoom.tsx +194 -0
- package/src/ui/SetupWizard.tsx +82 -0
- package/src/ws/client.ts +182 -0
- package/tsconfig.json +17 -0
- package/tsup.config.ts +12 -0
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"]}
|