freertc 0.1.17 → 0.1.19

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/README.md CHANGED
@@ -216,7 +216,7 @@ Quick checks:
216
216
  Expected `/health` response includes JSON like:
217
217
 
218
218
  ```json
219
- {"ok":true,"version":"0.1.17","protocol_version":"1.0","peers":0}
219
+ {"ok":true,"version":"0.1.19","protocol_version":"1.0","peers":0}
220
220
  ```
221
221
 
222
222
  ## Auto WebRTC two-tab test
package/bin/freertc.mjs CHANGED
@@ -102,6 +102,11 @@ function extractRouteHost(configText) {
102
102
  return normalizeHost(pattern.split('/')[0]);
103
103
  }
104
104
 
105
+ function extractCurrentWorkerName(configText) {
106
+ const match = configText.match(/"name"\s*:\s*"([^"]+)"/);
107
+ return match?.[1] || null;
108
+ }
109
+
105
110
  function extractRelayUrlHost(configText) {
106
111
  const relayUrlMatch = configText.match(/"RELAY_URL"\s*:\s*"([^"]*)"/);
107
112
  if (!relayUrlMatch) return null;
@@ -201,6 +206,131 @@ function patchWranglerScriptName(configPath, workerName) {
201
206
  return true;
202
207
  }
203
208
 
209
+ function getApiToken() {
210
+ return process.env.CLOUDFLARE_API_TOKEN || process.env.CF_API_TOKEN || '';
211
+ }
212
+
213
+ function getAccountId() {
214
+ return process.env.CLOUDFLARE_ACCOUNT_ID || process.env.CF_ACCOUNT_ID || '';
215
+ }
216
+
217
+ function guessZoneCandidates(host) {
218
+ const normalized = normalizeHost(host);
219
+ if (!normalized) return [];
220
+ const parts = normalized.split('.');
221
+ const candidates = [];
222
+ for (let i = 0; i <= Math.max(0, parts.length - 2); i += 1) {
223
+ candidates.push(parts.slice(i).join('.'));
224
+ }
225
+ return Array.from(new Set(candidates));
226
+ }
227
+
228
+ function routePatternMatchesHost(pattern, host) {
229
+ const normalizedHost = normalizeHost(host);
230
+ if (!pattern || !normalizedHost) return false;
231
+ const rawHost = normalizeHost(String(pattern).split('/')[0]);
232
+ if (!rawHost) return false;
233
+ if (rawHost === normalizedHost) return true;
234
+ if (rawHost.startsWith('*.')) {
235
+ const suffix = rawHost.slice(2);
236
+ return normalizedHost === suffix || normalizedHost.endsWith(`.${suffix}`);
237
+ }
238
+ return false;
239
+ }
240
+
241
+ async function apiGetJson(url, token) {
242
+ const response = await fetch(url, {
243
+ headers: {
244
+ Authorization: `Bearer ${token}`,
245
+ Accept: 'application/json'
246
+ }
247
+ });
248
+ if (!response.ok) {
249
+ return null;
250
+ }
251
+
252
+ let payload;
253
+ try {
254
+ payload = await response.json();
255
+ } catch {
256
+ return null;
257
+ }
258
+
259
+ if (!payload?.success) {
260
+ return null;
261
+ }
262
+
263
+ return payload.result;
264
+ }
265
+
266
+ async function resolveZoneIdForHost(host, token) {
267
+ const candidates = guessZoneCandidates(host);
268
+ for (const candidate of candidates) {
269
+ const result = await apiGetJson(`https://api.cloudflare.com/client/v4/zones?name=${encodeURIComponent(candidate)}`, token);
270
+ if (Array.isArray(result) && result.length > 0 && result[0]?.id) {
271
+ return result[0].id;
272
+ }
273
+ }
274
+ return null;
275
+ }
276
+
277
+ async function findRouteOwnerForHost(host) {
278
+ const token = getApiToken();
279
+ const accountId = getAccountId();
280
+ if (!token || !accountId) {
281
+ return null;
282
+ }
283
+
284
+ const zoneId = await resolveZoneIdForHost(host, token);
285
+ if (!zoneId) {
286
+ return null;
287
+ }
288
+
289
+ const routes = await apiGetJson(`https://api.cloudflare.com/client/v4/zones/${zoneId}/workers/routes`, token);
290
+ if (!Array.isArray(routes)) {
291
+ return null;
292
+ }
293
+
294
+ // Prefer exact host matches over wildcard matches.
295
+ const exact = routes.find((route) => {
296
+ const patternHost = normalizeHost(String(route?.pattern || '').split('/')[0]);
297
+ const scriptName = route?.script || route?.script_name;
298
+ return patternHost === normalizeHost(host) && typeof scriptName === 'string' && scriptName.trim();
299
+ });
300
+ if (exact?.script || exact?.script_name) {
301
+ return exact.script || exact.script_name;
302
+ }
303
+
304
+ const wildcard = routes.find((route) => {
305
+ const scriptName = route?.script || route?.script_name;
306
+ return routePatternMatchesHost(route?.pattern, host) && typeof scriptName === 'string' && scriptName.trim();
307
+ });
308
+ return wildcard ? (wildcard.script || wildcard.script_name) : null;
309
+ }
310
+
311
+ async function maybeAlignWorkerNameToRouteOwner(configPath) {
312
+ const text = fs.readFileSync(configPath, 'utf8');
313
+ const host = extractRouteHost(text) || extractRelayUrlHost(text) || extractDatabaseDomainHost(text);
314
+ if (!host) {
315
+ return;
316
+ }
317
+
318
+ const routeOwner = await findRouteOwnerForHost(host);
319
+ if (!routeOwner) {
320
+ return;
321
+ }
322
+
323
+ const currentName = extractCurrentWorkerName(text);
324
+ if (!currentName || currentName === routeOwner) {
325
+ return;
326
+ }
327
+
328
+ const patched = patchWranglerScriptName(configPath, routeOwner);
329
+ if (patched) {
330
+ console.log(`Detected route owner worker "${routeOwner}" for ${host}. Auto-updated wrangler.jsonc name before deploy.`);
331
+ }
332
+ }
333
+
204
334
  function runDeployWithAutoRouteRecovery(wrangler, configPath, extraArgs = []) {
205
335
  const attemptedOwners = new Set();
206
336
 
@@ -363,6 +493,7 @@ async function main() {
363
493
  ensureProjectFiles(PROJECT_ROOT);
364
494
  const configPath = requireWranglerConfig();
365
495
  autoPatchWranglerConfig(configPath);
496
+ await maybeAlignWorkerNameToRouteOwner(configPath);
366
497
  validateWranglerConfigForDeploy(configPath);
367
498
  const wrangler = resolveWranglerCommand(PROJECT_ROOT);
368
499
  runDeployWithAutoRouteRecovery(wrangler, configPath, rest);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "freertc",
3
- "version": "0.1.17",
3
+ "version": "0.1.19",
4
4
  "description": "Cloudflare Worker signaling relay for WebRTC peers with D1 storage.",
5
5
  "keywords": [
6
6
  "webrtc",
package/public/app.js CHANGED
@@ -1283,6 +1283,12 @@ createApp({
1283
1283
 
1284
1284
  function selectPeer(peerId, options = {}) {
1285
1285
  const { manual = false, allowUnannounced = false } = options;
1286
+
1287
+ if (peerId === fromPeer.value) {
1288
+ pushLog("discovery:error", "Ignoring self peer selection");
1289
+ toPeer.value = "";
1290
+ return;
1291
+ }
1286
1292
 
1287
1293
  // Only allow selecting announced peers to prevent phantom links.
1288
1294
  const isAnnounced = discoveredPeers.value.some((p) => p?.peer_id === peerId);
@@ -1889,6 +1895,10 @@ createApp({
1889
1895
  return;
1890
1896
  }
1891
1897
 
1898
+ if (message.from && message.from === fromPeer.value) {
1899
+ return;
1900
+ }
1901
+
1892
1902
  if (message.type === "error") {
1893
1903
  pushLog("server:error", message.body || {});
1894
1904
  return;
@@ -2087,6 +2097,9 @@ createApp({
2087
2097
  }
2088
2098
 
2089
2099
  const peerId = message.from;
2100
+ if (peerId === fromPeer.value) {
2101
+ return;
2102
+ }
2090
2103
  const link = getMeshLink(peerId);
2091
2104
 
2092
2105
  if (!hasMeshCapacityForPeer(peerId)) {
@@ -2144,6 +2157,9 @@ createApp({
2144
2157
  return;
2145
2158
  }
2146
2159
  const peerId = message.from;
2160
+ if (peerId === fromPeer.value) {
2161
+ return;
2162
+ }
2147
2163
  const link = getMeshLink(peerId);
2148
2164
 
2149
2165
  if (link.dc && link.dc.readyState === "open") {
@@ -2176,6 +2192,9 @@ createApp({
2176
2192
  return;
2177
2193
  }
2178
2194
  const peerId = message.from;
2195
+ if (peerId === fromPeer.value) {
2196
+ return;
2197
+ }
2179
2198
  const rejectCode = message.body?.code || "connect_reject";
2180
2199
  const link = getMeshLink(peerId);
2181
2200
  link.connectRequested = false;
@@ -2205,6 +2224,9 @@ createApp({
2205
2224
  return;
2206
2225
  }
2207
2226
  const peerId = message.from;
2227
+ if (peerId === fromPeer.value) {
2228
+ return;
2229
+ }
2208
2230
 
2209
2231
  if (!hasMeshCapacityForPeer(peerId)) {
2210
2232
  sendRelayEnvelope(
@@ -2288,6 +2310,9 @@ createApp({
2288
2310
  return;
2289
2311
  }
2290
2312
  const peerId = message.from;
2313
+ if (peerId === fromPeer.value) {
2314
+ return;
2315
+ }
2291
2316
  const link = getMeshLink(peerId, false);
2292
2317
  if (!link?.pc || link.phase === "connected") {
2293
2318
  return;
@@ -2331,6 +2356,10 @@ createApp({
2331
2356
  return;
2332
2357
  }
2333
2358
 
2359
+ if (message.from === fromPeer.value) {
2360
+ return;
2361
+ }
2362
+
2334
2363
  const link = getMeshLink(message.from);
2335
2364
  if (!Array.isArray(link.pendingCandidates)) {
2336
2365
  link.pendingCandidates = [];
@@ -7,7 +7,7 @@ import { fileURLToPath } from 'node:url';
7
7
  import { WebSocketServer } from 'ws';
8
8
 
9
9
  const PSP_VERSION = '1.0';
10
- const WORKER_VERSION = '0.1.17';
10
+ const WORKER_VERSION = '0.1.19';
11
11
  const DEFAULT_TTL_MS = 30_000;
12
12
  const MAX_TTL_MS = 120_000;
13
13
  const MAX_MESSAGE_SIZE = 64 * 1024;
package/src/index.js CHANGED
@@ -1,5 +1,5 @@
1
1
  const PSP_VERSION = "1.0";
2
- const WORKER_VERSION = "0.1.17";
2
+ const WORKER_VERSION = "0.1.19";
3
3
 
4
4
  const DISCOVERY_TYPES = new Set(["announce", "withdraw", "discover", "peer_list", "redirect"]);
5
5
  const NEGOTIATION_TYPES = new Set(["connect_request", "connect_accept", "connect_reject", "offer", "answer", "ice_candidate", "ice_end", "renegotiate"]);
@@ -554,6 +554,11 @@ async function handleClientMessage(socket, rawData, env, ctx, prevPeerKey = null
554
554
  const db = env.DB;
555
555
  const peerKey = `${network}:${peerId}`;
556
556
 
557
+ // Never relay or process peer-to-peer negotiation from a peer to itself.
558
+ if (message.to && message.to === peerId && RELAY_TYPES.has(type)) {
559
+ return { peerKey, network, peerId };
560
+ }
561
+
557
562
  // Subscribe to network on first message and whenever network changes on the same socket.
558
563
  if (!prevPeerKey || prevNetwork !== network) {
559
564
  if (prevNetwork && prevNetwork !== network) {
@@ -660,6 +665,7 @@ async function handleClientMessage(socket, rawData, env, ctx, prevPeerKey = null
660
665
  } else if (RELAY_TYPES.has(type)) {
661
666
  // RTC negotiation messages - relay immediately if online, queue if offline
662
667
  if (!message.to) return { peerKey, network, peerId };
668
+ if (message.to === peerId) return { peerKey, network, peerId };
663
669
 
664
670
  // Try immediate delivery to live peer
665
671
  const liveKey = `${network}:${message.to}`;