@fedify/fedify 2.3.0-dev.1079 → 2.3.0-dev.1099
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/{builder-DqT-OPZ8.mjs → builder-BkRRjxzb.mjs} +2 -2
- package/dist/{chunk-nlSIicah.js → chunk-CRNNMoPX.js} +2 -2
- package/dist/{client-z-8dc-e1.d.cts → client-CAM_bQXx.d.cts} +1 -0
- package/dist/{client-AtlibPOU.d.ts → client-CSddvgWN.d.ts} +1 -2
- package/dist/compat/mod.d.cts +2 -1
- package/dist/compat/mod.d.ts +2 -3
- package/dist/compat/mod.js +3 -3
- package/dist/compat/transformers.test.mjs +1 -1
- package/dist/{context-DrNqYkPw.d.ts → context-C0C_sRha.d.cts} +6 -7
- package/dist/{context-BKLGj9QO.d.cts → context-Dqgt8saU.d.ts} +6 -5
- package/dist/{deno-BxqkYxIf.mjs → deno-DBabeupC.mjs} +1 -1
- package/dist/{docloader-vsZP01ww.mjs → docloader-DA5FzJOR.mjs} +2 -2
- package/dist/federation/builder.test.mjs +1 -1
- package/dist/federation/handler.test.mjs +84 -2
- package/dist/federation/idempotency.test.mjs +2 -2
- package/dist/federation/middleware.test.mjs +383 -16
- package/dist/federation/mod.cjs +1 -1
- package/dist/federation/mod.d.cts +4 -3
- package/dist/federation/mod.d.ts +4 -5
- package/dist/federation/mod.js +2 -2
- package/dist/federation/send.test.mjs +3 -3
- package/dist/federation/webfinger.test.mjs +1 -1
- package/dist/{http-DM4ZAFCe.mjs → http-5G18W3NP.mjs} +58 -10
- package/dist/{http-CrGuipxe.d.cts → http-BDZeS5om.d.ts} +2 -1
- package/dist/{http-aQzN9Ayi.d.ts → http-C87EWkO0.d.cts} +2 -3
- package/dist/{http-b__OS_rJ.js → http-Dzy5c472.js} +58 -10
- package/dist/{http-DQ25_ruv.cjs → http-W2u_KBoQ.cjs} +57 -9
- package/dist/{key-BNp_b9oE.mjs → key-D9dUsyow.mjs} +1 -1
- package/dist/{kv-CbLNp3zQ.d.cts → kv-D6hNiMTK.d.ts} +1 -0
- package/dist/{kv-cache-DzctboPD.cjs → kv-cache-BygrlQ1c.cjs} +2 -2
- package/dist/{kv-cache-D5vjOi5y.js → kv-cache-CBSgxEsZ.js} +2 -2
- package/dist/{kv-cache-EaVpV2xQ.mjs → kv-cache-CiiNwT6W.mjs} +1 -1
- package/dist/{kv-GFYnFoOl.d.ts → kv-gJ8LYbxX.d.cts} +1 -3
- package/dist/{ld-L9w529xq.mjs → ld-hbxDLO1k.mjs} +2 -2
- package/dist/{middleware-CTdNwf_s.mjs → middleware-BXnhAGF9.mjs} +188 -114
- package/dist/{middleware-CMF242Rg.cjs → middleware-Caj827xW.cjs} +305 -107
- package/dist/{middleware-CpCSD43m.mjs → middleware-DZQsPMZb.mjs} +1 -1
- package/dist/{middleware-DZNHpEbh.js → middleware-vCF_cKAq.js} +309 -111
- package/dist/{mod-CLgIXe9w.d.ts → mod-B0rWmfW5.d.cts} +4 -5
- package/dist/{mod-CMEbIaNh.d.cts → mod-BhU_H1I_.d.ts} +4 -3
- package/dist/{mod-B8Z8mBLk.d.ts → mod-CLPnQPsv.d.cts} +2 -3
- package/dist/{mod-Cr3f-ACa.d.cts → mod-DHO9lk3D.d.ts} +3 -2
- package/dist/{mod-CR8soWa9.d.ts → mod-DXY9JF28.d.cts} +3 -4
- package/dist/{mod-DClCOv0M.d.cts → mod-Dx3-hqyo.d.ts} +2 -1
- package/dist/mod.cjs +4 -4
- package/dist/mod.d.cts +9 -8
- package/dist/mod.d.ts +9 -10
- package/dist/mod.js +9 -9
- package/dist/nodeinfo/handler.test.mjs +1 -1
- package/dist/nodeinfo/mod.d.cts +2 -1
- package/dist/nodeinfo/mod.d.ts +2 -3
- package/dist/nodeinfo/mod.js +3 -3
- package/dist/otel/mod.d.cts +2 -1
- package/dist/otel/mod.d.ts +2 -3
- package/dist/otel/mod.js +2 -2
- package/dist/{owner-CptqhsOy.d.cts → owner-CnngXDNJ.d.ts} +2 -1
- package/dist/{owner-74ARJ5TL.d.ts → owner-DEvZuyOE.d.cts} +2 -3
- package/dist/{owner-C30LGgMz.mjs → owner-DwJe0BH9.mjs} +2 -2
- package/dist/{proof-zrLeLWgt.cjs → proof-CZCaAURh.cjs} +1 -1
- package/dist/{proof-CO1qAbN8.js → proof-DMJJZnKd.js} +2 -2
- package/dist/{proof-BjToRsXF.mjs → proof-erpV_J_n.mjs} +2 -2
- package/dist/runtime/mod.d.cts +1 -0
- package/dist/runtime/mod.d.ts +1 -2
- package/dist/runtime/mod.js +3 -3
- package/dist/{send-DzbMznU6.mjs → send-BOwz4Hw5.mjs} +127 -3
- package/dist/sig/http.test.mjs +120 -2
- package/dist/sig/key.test.mjs +1 -1
- package/dist/sig/ld.test.mjs +2 -2
- package/dist/sig/mod.cjs +2 -2
- package/dist/sig/mod.d.cts +4 -3
- package/dist/sig/mod.d.ts +4 -5
- package/dist/sig/mod.js +4 -4
- package/dist/sig/owner.test.mjs +1 -1
- package/dist/sig/proof.test.mjs +1 -1
- package/dist/{transformers-ve6e2xcg.js → transformers-BGMIq1cs.js} +2 -2
- package/dist/{types-hvL8ElAs.js → types-CAY3OdLq.js} +2 -2
- package/dist/utils/docloader.test.mjs +2 -2
- package/dist/utils/kv-cache.test.mjs +1 -1
- package/dist/utils/mod.cjs +1 -1
- package/dist/utils/mod.d.cts +2 -1
- package/dist/utils/mod.d.ts +2 -3
- package/dist/utils/mod.js +3 -3
- package/dist/vocab/mod.d.cts +1 -0
- package/dist/vocab/mod.d.ts +1 -2
- package/dist/vocab/mod.js +2 -2
- package/package.json +6 -6
|
@@ -11,17 +11,18 @@ import { t as assertNotEquals } from "../assert_not_equals--wG9hV7u.mjs";
|
|
|
11
11
|
import { t as assertStrictEquals } from "../assert_strict_equals-Dmjbg-bA.mjs";
|
|
12
12
|
import { t as assert } from "../assert-DikXweDx.mjs";
|
|
13
13
|
import { t as esm_default } from "../esm-sdtqOUPu.mjs";
|
|
14
|
-
import { l as verifyRequest, s as signRequest } from "../http-
|
|
14
|
+
import { l as verifyRequest, s as signRequest } from "../http-5G18W3NP.mjs";
|
|
15
15
|
import { a as rsaPrivateKey3, c as rsaPublicKey3, i as rsaPrivateKey2, n as ed25519PrivateKey, r as ed25519PublicKey, s as rsaPublicKey2, t as ed25519Multikey } from "../keys-C3kae-6B.mjs";
|
|
16
|
-
import { t as getAuthenticatedDocumentLoader } from "../docloader-
|
|
17
|
-
import { a as signJsonLd, o as verifyJsonLd, r as detachSignature } from "../ld-
|
|
18
|
-
import { t as doesActorOwnKey } from "../owner-
|
|
19
|
-
import { i as verifyObject, r as signObject } from "../proof-
|
|
16
|
+
import { t as getAuthenticatedDocumentLoader } from "../docloader-DA5FzJOR.mjs";
|
|
17
|
+
import { a as signJsonLd, o as verifyJsonLd, r as detachSignature } from "../ld-hbxDLO1k.mjs";
|
|
18
|
+
import { t as doesActorOwnKey } from "../owner-DwJe0BH9.mjs";
|
|
19
|
+
import { i as verifyObject, r as signObject } from "../proof-erpV_J_n.mjs";
|
|
20
20
|
import { t as MemoryKvStore } from "../kv-x2IvBUyq.mjs";
|
|
21
|
-
import { i as KvSpecDeterminer, n as FederationImpl, o as createFederation, r as InboxContextImpl, t as ContextImpl } from "../middleware-
|
|
21
|
+
import { i as KvSpecDeterminer, n as FederationImpl, o as createFederation, r as InboxContextImpl, t as ContextImpl } from "../middleware-BXnhAGF9.mjs";
|
|
22
22
|
import { configure, reset } from "@logtape/logtape";
|
|
23
23
|
import * as vocab from "@fedify/vocab";
|
|
24
24
|
import { getTypeId, lookupObject } from "@fedify/vocab";
|
|
25
|
+
import { SpanStatusCode } from "@opentelemetry/api";
|
|
25
26
|
import { createTestMeterProvider, createTestTracerProvider, mockDocumentLoader, test } from "@fedify/fixture";
|
|
26
27
|
import { FetchError, getDocumentLoader } from "@fedify/vocab-runtime";
|
|
27
28
|
import serialize from "json-canon";
|
|
@@ -1918,19 +1919,22 @@ test("Federation.setOutboxListeners()", async (t) => {
|
|
|
1918
1919
|
};
|
|
1919
1920
|
let listenCalled = false;
|
|
1920
1921
|
const enqueued = [];
|
|
1922
|
+
const queue = {
|
|
1923
|
+
enqueue(message) {
|
|
1924
|
+
enqueued.push(message);
|
|
1925
|
+
return Promise.resolve();
|
|
1926
|
+
},
|
|
1927
|
+
listen() {
|
|
1928
|
+
listenCalled = true;
|
|
1929
|
+
return Promise.resolve();
|
|
1930
|
+
}
|
|
1931
|
+
};
|
|
1932
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
1921
1933
|
const federation = new FederationImpl({
|
|
1922
1934
|
kv,
|
|
1923
1935
|
contextLoaderFactory: () => mockDocumentLoader,
|
|
1924
|
-
|
|
1925
|
-
|
|
1926
|
-
enqueued.push(message);
|
|
1927
|
-
return Promise.resolve();
|
|
1928
|
-
},
|
|
1929
|
-
listen() {
|
|
1930
|
-
listenCalled = true;
|
|
1931
|
-
return Promise.resolve();
|
|
1932
|
-
}
|
|
1933
|
-
}
|
|
1936
|
+
meterProvider,
|
|
1937
|
+
queue
|
|
1934
1938
|
});
|
|
1935
1939
|
federation.setActorDispatcher("/users/{identifier}", (_ctx, identifier) => identifier === "john" ? new vocab.Person({}) : null).setKeyPairsDispatcher(() => [{
|
|
1936
1940
|
privateKey: rsaPrivateKey2,
|
|
@@ -1956,6 +1960,10 @@ test("Federation.setOutboxListeners()", async (t) => {
|
|
|
1956
1960
|
assertEquals(enqueued.length, 1);
|
|
1957
1961
|
assertEquals(enqueued[0].type, "outbox");
|
|
1958
1962
|
assertEquals(enqueued[0].actorIds, ["https://remote.example/users/alice"]);
|
|
1963
|
+
const enqueuedMetrics = recorder.getMeasurements("fedify.queue.task.enqueued");
|
|
1964
|
+
assertEquals(enqueuedMetrics.length, 1);
|
|
1965
|
+
assertEquals(enqueuedMetrics[0].attributes["fedify.queue.role"], "outbox");
|
|
1966
|
+
assertEquals(enqueuedMetrics[0].attributes["fedify.queue.task.attempt"], 0);
|
|
1959
1967
|
});
|
|
1960
1968
|
});
|
|
1961
1969
|
test("Federation.setInboxDispatcher()", async (t) => {
|
|
@@ -2321,6 +2329,28 @@ test("FederationImpl.processQueuedTask()", async (t) => {
|
|
|
2321
2329
|
const durations = recorder.getMeasurements("activitypub.inbox.processing_duration");
|
|
2322
2330
|
assertEquals(durations.length, 1);
|
|
2323
2331
|
assertEquals(durations[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2332
|
+
const started = recorder.getMeasurements("fedify.queue.task.started");
|
|
2333
|
+
assertEquals(started.length, 1);
|
|
2334
|
+
assertEquals(started[0].attributes["fedify.queue.role"], "inbox");
|
|
2335
|
+
const completed = recorder.getMeasurements("fedify.queue.task.completed");
|
|
2336
|
+
assertEquals(completed.length, 1);
|
|
2337
|
+
assertEquals(completed[0].attributes["fedify.queue.role"], "inbox");
|
|
2338
|
+
assertEquals(completed[0].attributes["fedify.queue.task.result"], "completed");
|
|
2339
|
+
assertEquals(completed[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2340
|
+
assertEquals(recorder.getMeasurements("fedify.queue.task.failed").length, 0);
|
|
2341
|
+
const taskDurations = recorder.getMeasurements("fedify.queue.task.duration");
|
|
2342
|
+
assertEquals(taskDurations.length, 1);
|
|
2343
|
+
assertEquals(taskDurations[0].type, "histogram");
|
|
2344
|
+
assertEquals(taskDurations[0].attributes["fedify.queue.role"], "inbox");
|
|
2345
|
+
assertEquals(taskDurations[0].attributes["fedify.queue.task.result"], "completed");
|
|
2346
|
+
const inFlight = recorder.getMeasurements("fedify.queue.task.in_flight");
|
|
2347
|
+
assertEquals(inFlight.length, 2);
|
|
2348
|
+
assertEquals(inFlight[0].type, "upDownCounter");
|
|
2349
|
+
assertEquals(inFlight[0].value, 1);
|
|
2350
|
+
assertEquals(inFlight[1].value, -1);
|
|
2351
|
+
assertEquals(inFlight[0].attributes, inFlight[1].attributes);
|
|
2352
|
+
assertEquals(inFlight[0].attributes["fedify.queue.role"], "inbox");
|
|
2353
|
+
assertEquals(inFlight[0].attributes["activitypub.activity.type"], void 0);
|
|
2324
2354
|
});
|
|
2325
2355
|
});
|
|
2326
2356
|
test("FederationImpl.processQueuedTask() permanent failure", async (t) => {
|
|
@@ -2530,6 +2560,304 @@ test("FederationImpl.processQueuedTask() permanent failure", async (t) => {
|
|
|
2530
2560
|
});
|
|
2531
2561
|
esm_default.hardReset();
|
|
2532
2562
|
});
|
|
2563
|
+
test("FederationImpl.processQueuedTask() queue task metrics", async (t) => {
|
|
2564
|
+
await t.step("records failed result when worker re-throws (nativeRetrial)", async () => {
|
|
2565
|
+
const kv = new MemoryKvStore();
|
|
2566
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2567
|
+
const federation = new FederationImpl({
|
|
2568
|
+
kv,
|
|
2569
|
+
meterProvider,
|
|
2570
|
+
queue: {
|
|
2571
|
+
nativeRetrial: true,
|
|
2572
|
+
enqueue(_message, _options) {
|
|
2573
|
+
return Promise.resolve();
|
|
2574
|
+
},
|
|
2575
|
+
listen(_handler, _options) {
|
|
2576
|
+
return Promise.resolve();
|
|
2577
|
+
}
|
|
2578
|
+
}
|
|
2579
|
+
});
|
|
2580
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox").on(vocab.Create, () => {
|
|
2581
|
+
throw new Error("Intended error for testing");
|
|
2582
|
+
});
|
|
2583
|
+
await assertRejects(() => federation.processQueuedTask(void 0, {
|
|
2584
|
+
type: "inbox",
|
|
2585
|
+
id: crypto.randomUUID(),
|
|
2586
|
+
baseUrl: "https://example.com",
|
|
2587
|
+
activity: {
|
|
2588
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2589
|
+
type: "Create",
|
|
2590
|
+
id: "https://remote.example/activities/2",
|
|
2591
|
+
actor: "https://remote.example/users/alice",
|
|
2592
|
+
object: {
|
|
2593
|
+
type: "Note",
|
|
2594
|
+
content: "Hello world"
|
|
2595
|
+
}
|
|
2596
|
+
},
|
|
2597
|
+
started: (/* @__PURE__ */ new Date()).toISOString(),
|
|
2598
|
+
attempt: 0,
|
|
2599
|
+
identifier: null,
|
|
2600
|
+
traceContext: {}
|
|
2601
|
+
}), Error);
|
|
2602
|
+
assertEquals(recorder.getMeasurements("fedify.queue.task.completed").length, 0);
|
|
2603
|
+
const failed = recorder.getMeasurements("fedify.queue.task.failed");
|
|
2604
|
+
assertEquals(failed.length, 1);
|
|
2605
|
+
assertEquals(failed[0].attributes["fedify.queue.role"], "inbox");
|
|
2606
|
+
assertEquals(failed[0].attributes["fedify.queue.task.result"], "failed");
|
|
2607
|
+
assertEquals(failed[0].attributes["fedify.queue.native_retrial"], true);
|
|
2608
|
+
assertEquals(failed[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2609
|
+
const taskDurations = recorder.getMeasurements("fedify.queue.task.duration");
|
|
2610
|
+
assertEquals(taskDurations.length, 1);
|
|
2611
|
+
assertEquals(taskDurations[0].attributes["fedify.queue.task.result"], "failed");
|
|
2612
|
+
const inFlight = recorder.getMeasurements("fedify.queue.task.in_flight");
|
|
2613
|
+
assertEquals(inFlight.length, 2);
|
|
2614
|
+
assertEquals(inFlight[0].value, 1);
|
|
2615
|
+
assertEquals(inFlight[1].value, -1);
|
|
2616
|
+
assertEquals(inFlight[0].attributes, inFlight[1].attributes);
|
|
2617
|
+
});
|
|
2618
|
+
await t.step("records completed when retry handler swallows listener error", async () => {
|
|
2619
|
+
const kv = new MemoryKvStore();
|
|
2620
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2621
|
+
const queuedMessages = [];
|
|
2622
|
+
const federation = new FederationImpl({
|
|
2623
|
+
kv,
|
|
2624
|
+
meterProvider,
|
|
2625
|
+
queue: {
|
|
2626
|
+
enqueue(message, _options) {
|
|
2627
|
+
queuedMessages.push(message);
|
|
2628
|
+
return Promise.resolve();
|
|
2629
|
+
},
|
|
2630
|
+
listen(_handler, _options) {
|
|
2631
|
+
return Promise.resolve();
|
|
2632
|
+
}
|
|
2633
|
+
}
|
|
2634
|
+
});
|
|
2635
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox").on(vocab.Create, () => {
|
|
2636
|
+
throw new Error("Intended error for testing");
|
|
2637
|
+
});
|
|
2638
|
+
await federation.processQueuedTask(void 0, {
|
|
2639
|
+
type: "inbox",
|
|
2640
|
+
id: crypto.randomUUID(),
|
|
2641
|
+
baseUrl: "https://example.com",
|
|
2642
|
+
activity: {
|
|
2643
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2644
|
+
type: "Create",
|
|
2645
|
+
id: "https://remote.example/activities/retry",
|
|
2646
|
+
actor: "https://remote.example/users/alice",
|
|
2647
|
+
object: {
|
|
2648
|
+
type: "Note",
|
|
2649
|
+
content: "Hello world"
|
|
2650
|
+
}
|
|
2651
|
+
},
|
|
2652
|
+
started: (/* @__PURE__ */ new Date()).toISOString(),
|
|
2653
|
+
attempt: 0,
|
|
2654
|
+
identifier: null,
|
|
2655
|
+
traceContext: {}
|
|
2656
|
+
});
|
|
2657
|
+
assertEquals(queuedMessages.length, 1);
|
|
2658
|
+
const completed = recorder.getMeasurements("fedify.queue.task.completed");
|
|
2659
|
+
assertEquals(completed.length, 1);
|
|
2660
|
+
assertEquals(completed[0].attributes["fedify.queue.task.result"], "completed");
|
|
2661
|
+
const enqueued = recorder.getMeasurements("fedify.queue.task.enqueued");
|
|
2662
|
+
assertEquals(enqueued.length, 1);
|
|
2663
|
+
assertEquals(enqueued[0].attributes["fedify.queue.role"], "inbox");
|
|
2664
|
+
assertEquals(enqueued[0].attributes["fedify.queue.task.attempt"], 1);
|
|
2665
|
+
assertEquals(enqueued[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2666
|
+
});
|
|
2667
|
+
await t.step("records aborted result when worker re-throws AbortError", async () => {
|
|
2668
|
+
const kv = new MemoryKvStore();
|
|
2669
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2670
|
+
const [tracerProvider, exporter] = createTestTracerProvider();
|
|
2671
|
+
const federation = new FederationImpl({
|
|
2672
|
+
kv,
|
|
2673
|
+
meterProvider,
|
|
2674
|
+
tracerProvider,
|
|
2675
|
+
queue: {
|
|
2676
|
+
nativeRetrial: true,
|
|
2677
|
+
enqueue(_message, _options) {
|
|
2678
|
+
return Promise.resolve();
|
|
2679
|
+
},
|
|
2680
|
+
listen(_handler, _options) {
|
|
2681
|
+
return Promise.resolve();
|
|
2682
|
+
}
|
|
2683
|
+
}
|
|
2684
|
+
});
|
|
2685
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox").on(vocab.Create, () => {
|
|
2686
|
+
throw new DOMException("aborted", "AbortError");
|
|
2687
|
+
});
|
|
2688
|
+
await assertRejects(() => federation.processQueuedTask(void 0, {
|
|
2689
|
+
type: "inbox",
|
|
2690
|
+
id: crypto.randomUUID(),
|
|
2691
|
+
baseUrl: "https://example.com",
|
|
2692
|
+
activity: {
|
|
2693
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2694
|
+
type: "Create",
|
|
2695
|
+
id: "https://remote.example/activities/3",
|
|
2696
|
+
actor: "https://remote.example/users/alice",
|
|
2697
|
+
object: {
|
|
2698
|
+
type: "Note",
|
|
2699
|
+
content: "Hello world"
|
|
2700
|
+
}
|
|
2701
|
+
},
|
|
2702
|
+
started: (/* @__PURE__ */ new Date()).toISOString(),
|
|
2703
|
+
attempt: 0,
|
|
2704
|
+
identifier: null,
|
|
2705
|
+
traceContext: {}
|
|
2706
|
+
}), DOMException);
|
|
2707
|
+
assertEquals(recorder.getMeasurements("fedify.queue.task.failed").length, 0);
|
|
2708
|
+
assertEquals(recorder.getMeasurements("fedify.queue.task.completed").length, 0);
|
|
2709
|
+
const taskDurations = recorder.getMeasurements("fedify.queue.task.duration");
|
|
2710
|
+
assertEquals(taskDurations.length, 1);
|
|
2711
|
+
assertEquals(taskDurations[0].attributes["fedify.queue.task.result"], "aborted");
|
|
2712
|
+
const inboxSpans = exporter.getSpans("activitypub.inbox");
|
|
2713
|
+
assertEquals(inboxSpans.length, 1);
|
|
2714
|
+
assertEquals(inboxSpans[0].status.code, SpanStatusCode.UNSET);
|
|
2715
|
+
});
|
|
2716
|
+
await t.step("records native_retrial and backend attributes", async () => {
|
|
2717
|
+
const kv = new MemoryKvStore();
|
|
2718
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2719
|
+
class TestMessageQueue {
|
|
2720
|
+
nativeRetrial = true;
|
|
2721
|
+
enqueue(_message, _options) {
|
|
2722
|
+
return Promise.resolve();
|
|
2723
|
+
}
|
|
2724
|
+
listen(_handler, _options) {
|
|
2725
|
+
return Promise.resolve();
|
|
2726
|
+
}
|
|
2727
|
+
}
|
|
2728
|
+
const federation = new FederationImpl({
|
|
2729
|
+
kv,
|
|
2730
|
+
meterProvider,
|
|
2731
|
+
queue: new TestMessageQueue()
|
|
2732
|
+
});
|
|
2733
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox").on(vocab.Create, () => {});
|
|
2734
|
+
await federation.processQueuedTask(void 0, {
|
|
2735
|
+
type: "inbox",
|
|
2736
|
+
id: crypto.randomUUID(),
|
|
2737
|
+
baseUrl: "https://example.com",
|
|
2738
|
+
activity: {
|
|
2739
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2740
|
+
type: "Create",
|
|
2741
|
+
id: "https://remote.example/activities/4",
|
|
2742
|
+
actor: "https://remote.example/users/alice",
|
|
2743
|
+
object: {
|
|
2744
|
+
type: "Note",
|
|
2745
|
+
content: "Hello world"
|
|
2746
|
+
}
|
|
2747
|
+
},
|
|
2748
|
+
started: (/* @__PURE__ */ new Date()).toISOString(),
|
|
2749
|
+
attempt: 0,
|
|
2750
|
+
identifier: null,
|
|
2751
|
+
traceContext: {}
|
|
2752
|
+
});
|
|
2753
|
+
const completed = recorder.getMeasurements("fedify.queue.task.completed");
|
|
2754
|
+
assertEquals(completed.length, 1);
|
|
2755
|
+
assertEquals(completed[0].attributes["fedify.queue.backend"], "TestMessageQueue");
|
|
2756
|
+
assertEquals(completed[0].attributes["fedify.queue.native_retrial"], true);
|
|
2757
|
+
});
|
|
2758
|
+
await t.step("records outbox worker metrics on successful delivery", async () => {
|
|
2759
|
+
const kv = new MemoryKvStore();
|
|
2760
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2761
|
+
const federation = new FederationImpl({
|
|
2762
|
+
kv,
|
|
2763
|
+
meterProvider,
|
|
2764
|
+
queue: {
|
|
2765
|
+
enqueue(_message, _options) {
|
|
2766
|
+
return Promise.resolve();
|
|
2767
|
+
},
|
|
2768
|
+
listen(_handler, _options) {
|
|
2769
|
+
return Promise.resolve();
|
|
2770
|
+
}
|
|
2771
|
+
}
|
|
2772
|
+
});
|
|
2773
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox");
|
|
2774
|
+
esm_default.spyGlobal();
|
|
2775
|
+
esm_default.post("https://remote.example/inbox", { status: 202 });
|
|
2776
|
+
try {
|
|
2777
|
+
await federation.processQueuedTask(void 0, {
|
|
2778
|
+
type: "outbox",
|
|
2779
|
+
id: crypto.randomUUID(),
|
|
2780
|
+
baseUrl: "https://example.com",
|
|
2781
|
+
keys: [],
|
|
2782
|
+
activity: {
|
|
2783
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2784
|
+
type: "Create",
|
|
2785
|
+
id: "https://example.com/activities/1",
|
|
2786
|
+
actor: "https://example.com/users/alice",
|
|
2787
|
+
object: {
|
|
2788
|
+
type: "Note",
|
|
2789
|
+
content: "test"
|
|
2790
|
+
}
|
|
2791
|
+
},
|
|
2792
|
+
activityType: "https://www.w3.org/ns/activitystreams#Create",
|
|
2793
|
+
inbox: "https://remote.example/inbox",
|
|
2794
|
+
sharedInbox: false,
|
|
2795
|
+
started: (/* @__PURE__ */ new Date()).toISOString(),
|
|
2796
|
+
attempt: 0,
|
|
2797
|
+
headers: {},
|
|
2798
|
+
traceContext: {}
|
|
2799
|
+
});
|
|
2800
|
+
} finally {
|
|
2801
|
+
esm_default.hardReset();
|
|
2802
|
+
}
|
|
2803
|
+
const started = recorder.getMeasurements("fedify.queue.task.started");
|
|
2804
|
+
assertEquals(started.length, 1);
|
|
2805
|
+
assertEquals(started[0].attributes["fedify.queue.role"], "outbox");
|
|
2806
|
+
assertEquals(started[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2807
|
+
const completed = recorder.getMeasurements("fedify.queue.task.completed");
|
|
2808
|
+
assertEquals(completed.length, 1);
|
|
2809
|
+
assertEquals(completed[0].attributes["fedify.queue.task.result"], "completed");
|
|
2810
|
+
assertEquals(recorder.getMeasurements("fedify.queue.task.enqueued").length, 0);
|
|
2811
|
+
});
|
|
2812
|
+
await t.step("records started/completed for a fanout task with no recipients", async () => {
|
|
2813
|
+
const kv = new MemoryKvStore();
|
|
2814
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
2815
|
+
const exportedKey = await crypto.subtle.exportKey("jwk", rsaPrivateKey3);
|
|
2816
|
+
const federation = new FederationImpl({
|
|
2817
|
+
kv,
|
|
2818
|
+
meterProvider,
|
|
2819
|
+
queue: {
|
|
2820
|
+
enqueue(_message, _options) {
|
|
2821
|
+
return Promise.resolve();
|
|
2822
|
+
},
|
|
2823
|
+
listen(_handler, _options) {
|
|
2824
|
+
return Promise.resolve();
|
|
2825
|
+
}
|
|
2826
|
+
}
|
|
2827
|
+
});
|
|
2828
|
+
federation.setInboxListeners("/users/{identifier}/inbox", "/inbox");
|
|
2829
|
+
await federation.processQueuedTask(void 0, {
|
|
2830
|
+
type: "fanout",
|
|
2831
|
+
id: crypto.randomUUID(),
|
|
2832
|
+
baseUrl: "https://example.com",
|
|
2833
|
+
keys: [{
|
|
2834
|
+
keyId: "https://example.com/users/alice#main-key",
|
|
2835
|
+
privateKey: exportedKey
|
|
2836
|
+
}],
|
|
2837
|
+
inboxes: {},
|
|
2838
|
+
activity: {
|
|
2839
|
+
"@context": "https://www.w3.org/ns/activitystreams",
|
|
2840
|
+
type: "Create",
|
|
2841
|
+
id: "https://example.com/activities/1",
|
|
2842
|
+
actor: "https://example.com/users/alice",
|
|
2843
|
+
object: {
|
|
2844
|
+
type: "Note",
|
|
2845
|
+
content: "test"
|
|
2846
|
+
}
|
|
2847
|
+
},
|
|
2848
|
+
activityType: "https://www.w3.org/ns/activitystreams#Create",
|
|
2849
|
+
traceContext: {}
|
|
2850
|
+
});
|
|
2851
|
+
const started = recorder.getMeasurements("fedify.queue.task.started");
|
|
2852
|
+
assertEquals(started.length, 1);
|
|
2853
|
+
assertEquals(started[0].attributes["fedify.queue.role"], "fanout");
|
|
2854
|
+
assertEquals(started[0].attributes["activitypub.activity.type"], "https://www.w3.org/ns/activitystreams#Create");
|
|
2855
|
+
const completed = recorder.getMeasurements("fedify.queue.task.completed");
|
|
2856
|
+
assertEquals(completed.length, 1);
|
|
2857
|
+
assertEquals(completed[0].attributes["fedify.queue.role"], "fanout");
|
|
2858
|
+
assertEquals(completed[0].attributes["fedify.queue.task.result"], "completed");
|
|
2859
|
+
});
|
|
2860
|
+
});
|
|
2533
2861
|
test("ContextImpl.lookupObject()", async (t) => {
|
|
2534
2862
|
esm_default.spyGlobal();
|
|
2535
2863
|
esm_default.get("begin:https://localhost/.well-known/webfinger", {
|
|
@@ -3158,6 +3486,45 @@ test({
|
|
|
3158
3486
|
]);
|
|
3159
3487
|
}
|
|
3160
3488
|
});
|
|
3489
|
+
test({
|
|
3490
|
+
name: "ContextImpl.routeActivity() forwards meterProvider to inbox enqueue",
|
|
3491
|
+
permissions: {
|
|
3492
|
+
env: true,
|
|
3493
|
+
read: true
|
|
3494
|
+
},
|
|
3495
|
+
async fn() {
|
|
3496
|
+
const [meterProvider, recorder] = createTestMeterProvider();
|
|
3497
|
+
const enqueued = [];
|
|
3498
|
+
const federation = new FederationImpl({
|
|
3499
|
+
kv: new MemoryKvStore(),
|
|
3500
|
+
meterProvider,
|
|
3501
|
+
queue: {
|
|
3502
|
+
enqueue(message) {
|
|
3503
|
+
enqueued.push(message);
|
|
3504
|
+
return Promise.resolve();
|
|
3505
|
+
},
|
|
3506
|
+
listen() {
|
|
3507
|
+
return Promise.resolve();
|
|
3508
|
+
}
|
|
3509
|
+
}
|
|
3510
|
+
});
|
|
3511
|
+
federation.setInboxListeners("/u/{identifier}/i", "/i");
|
|
3512
|
+
const ctx = new ContextImpl({
|
|
3513
|
+
url: new URL("https://example.com/"),
|
|
3514
|
+
federation,
|
|
3515
|
+
data: void 0,
|
|
3516
|
+
documentLoader: mockDocumentLoader,
|
|
3517
|
+
contextLoader: documentLoader
|
|
3518
|
+
});
|
|
3519
|
+
const signedOffer = await signObject(new vocab.Offer({ actor: new URL("https://example.com/person2") }), ed25519PrivateKey, ed25519Multikey.id);
|
|
3520
|
+
assert(await ctx.routeActivity(null, signedOffer));
|
|
3521
|
+
assertEquals(enqueued.length, 1);
|
|
3522
|
+
const enqueuedMetrics = recorder.getMeasurements("fedify.queue.task.enqueued");
|
|
3523
|
+
assertEquals(enqueuedMetrics.length, 1);
|
|
3524
|
+
assertEquals(enqueuedMetrics[0].attributes["fedify.queue.role"], "inbox");
|
|
3525
|
+
assertEquals(enqueuedMetrics[0].attributes["fedify.queue.task.attempt"], 0);
|
|
3526
|
+
}
|
|
3527
|
+
});
|
|
3161
3528
|
test("ContextImpl.getCollectionUri()", () => {
|
|
3162
3529
|
const federation = new FederationImpl({ kv: new MemoryKvStore() });
|
|
3163
3530
|
const base = "https://example.com";
|
package/dist/federation/mod.cjs
CHANGED
|
@@ -2,7 +2,7 @@ const { Temporal } = require("@js-temporal/polyfill");
|
|
|
2
2
|
const { URLPattern } = require("urlpattern-polyfill");
|
|
3
3
|
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
4
4
|
require("../chunk-DDcVe30Y.cjs");
|
|
5
|
-
const require_middleware = require("../middleware-
|
|
5
|
+
const require_middleware = require("../middleware-Caj827xW.cjs");
|
|
6
6
|
let es_toolkit = require("es-toolkit");
|
|
7
7
|
//#region src/federation/kv.ts
|
|
8
8
|
/**
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
|
|
2
|
-
import { a as
|
|
3
|
-
import { n as
|
|
1
|
+
/// <reference lib="esnext.temporal" />
|
|
2
|
+
import { $ as ParallelMessageQueue, A as FederationKvPrefixes, B as Router, C as IdempotencyKeyCallback, Ct as WebFingerLinksDispatcher, D as ObjectCallbackSetters, Dt as buildCollectionSynchronizationHeader, E as InboxListenerSetters, Et as PageItems, F as RetryContext, G as respondWithObject, H as RouterOptions, I as RetryPolicy, J as InProcessMessageQueueOptions, K as respondWithObjectIfAcceptable, L as createExponentialBackoffPolicy, M as FederationQueueOptions, N as createFederation, O as OutboxListenerSetters, Ot as digest, P as CreateExponentialBackoffPolicyOptions, Q as MessageQueueListenOptions, R as Message, S as FederationStartQueueOptions, St as UnverifiedActivityReason, T as InboxChallengePolicy, Tt as SenderKeyPair, U as RouterRouteResult, V as RouterError, W as RespondWithObjectOptions, X as MessageQueueDepth, Y as MessageQueue, Z as MessageQueueEnqueueOptions, _ as Federatable, _t as OutboxListener, a as GetSignedKeyOptions, at as CollectionCounter, b as FederationFetchOptions, bt as SharedInboxKeyDispatcher, c as ParseUriResult, ct as CustomCollectionCounter, d as SendActivityOptions, dt as InboxErrorHandler, et as ActorAliasMapper, f as SendActivityOptionsForCollection, ft as InboxListener, g as CustomCollectionCallbackSetters, gt as OutboxErrorHandler, h as ConstructorWithTypeId, ht as ObjectDispatcher, i as GetActorOptions, it as AuthorizePredicate, j as FederationOrigin, k as Rfc6570Expression, l as RequestContext, lt as CustomCollectionCursor, m as CollectionCallbackSetters, mt as ObjectAuthorizePredicate, n as Context, nt as ActorHandleMapper, o as InboxContext, ot as CollectionCursor, p as ActorCallbackSetters, pt as NodeInfoDispatcher, q as InProcessMessageQueue, r as ForwardActivityOptions, rt as ActorKeyPairsDispatcher, s as OutboxContext, st as CollectionDispatcher, t as ActorKeyPair, tt as ActorDispatcher, u as RouteActivityOptions, ut as CustomCollectionDispatcher, v as Federation, vt as OutboxListenerErrorHandler, w as IdempotencyStrategy, wt as SendActivityError, x as FederationOptions, xt as UnverifiedActivityHandler, y as FederationBuilder, yt as OutboxPermanentFailureHandler, z as createFederationBuilder } from "../context-C0C_sRha.cjs";
|
|
3
|
+
import { a as MemoryKvStore, i as KvStoreSetOptions, n as KvStore, r as KvStoreListEntry, t as KvKey } from "../kv-gJ8LYbxX.cjs";
|
|
4
|
+
import { n as handleWebFinger, t as WebFingerHandlerParameters } from "../mod-CLPnQPsv.cjs";
|
|
4
5
|
export { ActorAliasMapper, ActorCallbackSetters, ActorDispatcher, ActorHandleMapper, ActorKeyPair, ActorKeyPairsDispatcher, AuthorizePredicate, CollectionCallbackSetters, CollectionCounter, CollectionCursor, CollectionDispatcher, ConstructorWithTypeId, Context, CreateExponentialBackoffPolicyOptions, CustomCollectionCallbackSetters, CustomCollectionCounter, CustomCollectionCursor, CustomCollectionDispatcher, Federatable, Federation, FederationBuilder, FederationFetchOptions, FederationKvPrefixes, FederationOptions, FederationOrigin, FederationQueueOptions, FederationStartQueueOptions, ForwardActivityOptions, GetActorOptions, GetSignedKeyOptions, IdempotencyKeyCallback, IdempotencyStrategy, InProcessMessageQueue, InProcessMessageQueueOptions, InboxChallengePolicy, InboxContext, InboxErrorHandler, InboxListener, InboxListenerSetters, KvKey, KvStore, KvStoreListEntry, KvStoreSetOptions, MemoryKvStore, Message, MessageQueue, MessageQueueDepth, MessageQueueEnqueueOptions, MessageQueueListenOptions, NodeInfoDispatcher, ObjectAuthorizePredicate, ObjectCallbackSetters, ObjectDispatcher, OutboxContext, OutboxErrorHandler, OutboxListener, OutboxListenerErrorHandler, OutboxListenerSetters, OutboxPermanentFailureHandler, PageItems, ParallelMessageQueue, ParseUriResult, RequestContext, RespondWithObjectOptions, RetryContext, RetryPolicy, Rfc6570Expression, RouteActivityOptions, Router, RouterError, RouterOptions, RouterRouteResult, SendActivityError, SendActivityOptions, SendActivityOptionsForCollection, SenderKeyPair, SharedInboxKeyDispatcher, UnverifiedActivityHandler, UnverifiedActivityReason, WebFingerHandlerParameters, WebFingerLinksDispatcher, buildCollectionSynchronizationHeader, createExponentialBackoffPolicy, createFederation, createFederationBuilder, digest, handleWebFinger, respondWithObject, respondWithObjectIfAcceptable };
|
package/dist/federation/mod.d.ts
CHANGED
|
@@ -1,6 +1,5 @@
|
|
|
1
|
-
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
import { n as handleWebFinger, t as WebFingerHandlerParameters } from "../mod-B8Z8mBLk.js";
|
|
1
|
+
/// <reference lib="esnext.temporal" />
|
|
2
|
+
import { $ as ParallelMessageQueue, A as FederationKvPrefixes, B as Router, C as IdempotencyKeyCallback, Ct as WebFingerLinksDispatcher, D as ObjectCallbackSetters, Dt as buildCollectionSynchronizationHeader, E as InboxListenerSetters, Et as PageItems, F as RetryContext, G as respondWithObject, H as RouterOptions, I as RetryPolicy, J as InProcessMessageQueueOptions, K as respondWithObjectIfAcceptable, L as createExponentialBackoffPolicy, M as FederationQueueOptions, N as createFederation, O as OutboxListenerSetters, Ot as digest, P as CreateExponentialBackoffPolicyOptions, Q as MessageQueueListenOptions, R as Message, S as FederationStartQueueOptions, St as UnverifiedActivityReason, T as InboxChallengePolicy, Tt as SenderKeyPair, U as RouterRouteResult, V as RouterError, W as RespondWithObjectOptions, X as MessageQueueDepth, Y as MessageQueue, Z as MessageQueueEnqueueOptions, _ as Federatable, _t as OutboxListener, a as GetSignedKeyOptions, at as CollectionCounter, b as FederationFetchOptions, bt as SharedInboxKeyDispatcher, c as ParseUriResult, ct as CustomCollectionCounter, d as SendActivityOptions, dt as InboxErrorHandler, et as ActorAliasMapper, f as SendActivityOptionsForCollection, ft as InboxListener, g as CustomCollectionCallbackSetters, gt as OutboxErrorHandler, h as ConstructorWithTypeId, ht as ObjectDispatcher, i as GetActorOptions, it as AuthorizePredicate, j as FederationOrigin, k as Rfc6570Expression, l as RequestContext, lt as CustomCollectionCursor, m as CollectionCallbackSetters, mt as ObjectAuthorizePredicate, n as Context, nt as ActorHandleMapper, o as InboxContext, ot as CollectionCursor, p as ActorCallbackSetters, pt as NodeInfoDispatcher, q as InProcessMessageQueue, r as ForwardActivityOptions, rt as ActorKeyPairsDispatcher, s as OutboxContext, st as CollectionDispatcher, t as ActorKeyPair, tt as ActorDispatcher, u as RouteActivityOptions, ut as CustomCollectionDispatcher, v as Federation, vt as OutboxListenerErrorHandler, w as IdempotencyStrategy, wt as SendActivityError, x as FederationOptions, xt as UnverifiedActivityHandler, y as FederationBuilder, yt as OutboxPermanentFailureHandler, z as createFederationBuilder } from "../context-Dqgt8saU.js";
|
|
3
|
+
import { a as MemoryKvStore, i as KvStoreSetOptions, n as KvStore, r as KvStoreListEntry, t as KvKey } from "../kv-D6hNiMTK.js";
|
|
4
|
+
import { n as handleWebFinger, t as WebFingerHandlerParameters } from "../mod-Dx3-hqyo.js";
|
|
6
5
|
export { ActorAliasMapper, ActorCallbackSetters, ActorDispatcher, ActorHandleMapper, ActorKeyPair, ActorKeyPairsDispatcher, AuthorizePredicate, CollectionCallbackSetters, CollectionCounter, CollectionCursor, CollectionDispatcher, ConstructorWithTypeId, Context, CreateExponentialBackoffPolicyOptions, CustomCollectionCallbackSetters, CustomCollectionCounter, CustomCollectionCursor, CustomCollectionDispatcher, Federatable, Federation, FederationBuilder, FederationFetchOptions, FederationKvPrefixes, FederationOptions, FederationOrigin, FederationQueueOptions, FederationStartQueueOptions, ForwardActivityOptions, GetActorOptions, GetSignedKeyOptions, IdempotencyKeyCallback, IdempotencyStrategy, InProcessMessageQueue, InProcessMessageQueueOptions, InboxChallengePolicy, InboxContext, InboxErrorHandler, InboxListener, InboxListenerSetters, KvKey, KvStore, KvStoreListEntry, KvStoreSetOptions, MemoryKvStore, Message, MessageQueue, MessageQueueDepth, MessageQueueEnqueueOptions, MessageQueueListenOptions, NodeInfoDispatcher, ObjectAuthorizePredicate, ObjectCallbackSetters, ObjectDispatcher, OutboxContext, OutboxErrorHandler, OutboxListener, OutboxListenerErrorHandler, OutboxListenerSetters, OutboxPermanentFailureHandler, PageItems, ParallelMessageQueue, ParseUriResult, RequestContext, RespondWithObjectOptions, RetryContext, RetryPolicy, Rfc6570Expression, RouteActivityOptions, Router, RouterError, RouterOptions, RouterRouteResult, SendActivityError, SendActivityOptions, SendActivityOptionsForCollection, SenderKeyPair, SharedInboxKeyDispatcher, UnverifiedActivityHandler, UnverifiedActivityReason, WebFingerHandlerParameters, WebFingerLinksDispatcher, buildCollectionSynchronizationHeader, createExponentialBackoffPolicy, createFederation, createFederationBuilder, digest, handleWebFinger, respondWithObject, respondWithObjectIfAcceptable };
|
package/dist/federation/mod.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { Temporal } from "@js-temporal/polyfill";
|
|
2
|
-
import "urlpattern-polyfill";
|
|
3
|
-
import { a as createExponentialBackoffPolicy, c as buildCollectionSynchronizationHeader, d as Router, f as RouterError, i as SendActivityError, l as digest, o as respondWithObject, r as handleWebFinger, s as respondWithObjectIfAcceptable, t as createFederation, u as createFederationBuilder } from "../middleware-
|
|
2
|
+
import { URLPattern } from "urlpattern-polyfill";
|
|
3
|
+
import { a as createExponentialBackoffPolicy, c as buildCollectionSynchronizationHeader, d as Router, f as RouterError, i as SendActivityError, l as digest, o as respondWithObject, r as handleWebFinger, s as respondWithObjectIfAcceptable, t as createFederation, u as createFederationBuilder } from "../middleware-vCF_cKAq.js";
|
|
4
4
|
import { isEqual } from "es-toolkit";
|
|
5
5
|
//#region src/federation/kv.ts
|
|
6
6
|
/**
|
|
@@ -9,10 +9,10 @@ import { t as assertInstanceOf } from "../assert_instance_of-C4Ri6VuN.mjs";
|
|
|
9
9
|
import { t as assertNotEquals } from "../assert_not_equals--wG9hV7u.mjs";
|
|
10
10
|
import { t as assert } from "../assert-DikXweDx.mjs";
|
|
11
11
|
import { t as esm_default } from "../esm-sdtqOUPu.mjs";
|
|
12
|
-
import { l as verifyRequest } from "../http-
|
|
12
|
+
import { l as verifyRequest } from "../http-5G18W3NP.mjs";
|
|
13
13
|
import { i as rsaPrivateKey2, n as ed25519PrivateKey, s as rsaPublicKey2, t as ed25519Multikey } from "../keys-C3kae-6B.mjs";
|
|
14
|
-
import { t as doesActorOwnKey } from "../owner-
|
|
15
|
-
import { n as extractInboxes, r as sendActivity, t as SendActivityError } from "../send-
|
|
14
|
+
import { t as doesActorOwnKey } from "../owner-DwJe0BH9.mjs";
|
|
15
|
+
import { n as extractInboxes, r as sendActivity, t as SendActivityError } from "../send-BOwz4Hw5.mjs";
|
|
16
16
|
import { Activity, Application, Endpoints, Group, Person, Service } from "@fedify/vocab";
|
|
17
17
|
import { createTestMeterProvider, createTestTracerProvider, mockDocumentLoader, test } from "@fedify/fixture";
|
|
18
18
|
//#region ../../node_modules/.pnpm/@opentelemetry+sdk-metrics@2.7.1_@opentelemetry+api@1.9.1/node_modules/@opentelemetry/sdk-metrics/build/src/export/AggregationTemporality.js
|
|
@@ -5,7 +5,7 @@ import { r as createRequestContext } from "../context-BAE7AKLA.mjs";
|
|
|
5
5
|
import { t as assertEquals } from "../assert_equals-Ew3jOFa3.mjs";
|
|
6
6
|
import "../std__assert-CRDpx_HF.mjs";
|
|
7
7
|
import { t as MemoryKvStore } from "../kv-x2IvBUyq.mjs";
|
|
8
|
-
import { o as createFederation, s as handleWebFinger } from "../middleware-
|
|
8
|
+
import { o as createFederation, s as handleWebFinger } from "../middleware-BXnhAGF9.mjs";
|
|
9
9
|
import { Image, Link, Person, Tombstone } from "@fedify/vocab";
|
|
10
10
|
import { test } from "@fedify/fixture";
|
|
11
11
|
//#region src/federation/webfinger.test.ts
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { Temporal } from "@js-temporal/polyfill";
|
|
2
2
|
import "urlpattern-polyfill";
|
|
3
3
|
globalThis.addEventListener = () => {};
|
|
4
|
-
import { n as version, t as name } from "./deno-
|
|
4
|
+
import { n as version, t as name } from "./deno-DBabeupC.mjs";
|
|
5
5
|
import { i as validateAcceptSignature, n as fulfillAcceptSignature, r as parseAcceptSignature } from "./accept-CceiKpCy.mjs";
|
|
6
|
-
import { o as validateCryptoKey, r as fetchKeyDetailed } from "./key-
|
|
6
|
+
import { o as validateCryptoKey, r as fetchKeyDetailed } from "./key-D9dUsyow.mjs";
|
|
7
7
|
import { getLogger } from "@logtape/logtape";
|
|
8
8
|
import { CryptographicKey } from "@fedify/vocab";
|
|
9
9
|
import { SpanStatusCode, trace } from "@opentelemetry/api";
|
|
@@ -14,6 +14,7 @@ import { encodeHex } from "byte-encodings/hex";
|
|
|
14
14
|
import { Item, decodeDict, encodeItem } from "structured-field-values";
|
|
15
15
|
//#region src/sig/http.ts
|
|
16
16
|
const DEFAULT_MAX_REDIRECTION = 20;
|
|
17
|
+
const DOUBLE_KNOCK_TRANSPORT_RETRY_DELAY_MS = 100;
|
|
17
18
|
/**
|
|
18
19
|
* Signs a request using the given private key.
|
|
19
20
|
* @param request The request to sign.
|
|
@@ -842,6 +843,59 @@ function createRedirectRequest(request, location, body) {
|
|
|
842
843
|
cache: request.cache
|
|
843
844
|
});
|
|
844
845
|
}
|
|
846
|
+
async function fetchDoubleKnockRequest(request, signedRequest, signal) {
|
|
847
|
+
const maxAttempts = request.method === "GET" || request.method === "HEAD" ? 2 : 1;
|
|
848
|
+
for (let attempt = 1;; attempt++) try {
|
|
849
|
+
return await fetch(signedRequest, {
|
|
850
|
+
redirect: "manual",
|
|
851
|
+
signal
|
|
852
|
+
});
|
|
853
|
+
} catch (error) {
|
|
854
|
+
const abortedSignal = getAbortedSignal(signal, request.signal, signedRequest.signal);
|
|
855
|
+
if (abortedSignal != null) throw getAbortReason(abortedSignal);
|
|
856
|
+
if (isAbortError(error)) throw error;
|
|
857
|
+
if (attempt >= maxAttempts) throw createFetchError(request.url, error);
|
|
858
|
+
await sleep(DOUBLE_KNOCK_TRANSPORT_RETRY_DELAY_MS, signal, request.signal, signedRequest.signal);
|
|
859
|
+
}
|
|
860
|
+
}
|
|
861
|
+
function createFetchError(url, cause) {
|
|
862
|
+
const error = new FetchError(url, cause instanceof Error ? cause.message : String(cause));
|
|
863
|
+
error.cause = cause;
|
|
864
|
+
return error;
|
|
865
|
+
}
|
|
866
|
+
function isAbortError(error) {
|
|
867
|
+
return error instanceof Error && error.name === "AbortError";
|
|
868
|
+
}
|
|
869
|
+
async function sleep(ms, ...signals) {
|
|
870
|
+
const abortSignals = signals.filter((signal) => signal != null);
|
|
871
|
+
const abortedSignal = getAbortedSignal(...abortSignals);
|
|
872
|
+
if (abortedSignal != null) throw getAbortReason(abortedSignal);
|
|
873
|
+
if (abortSignals.length < 1) {
|
|
874
|
+
await new Promise((resolve) => setTimeout(resolve, ms));
|
|
875
|
+
return;
|
|
876
|
+
}
|
|
877
|
+
await new Promise((resolve, reject) => {
|
|
878
|
+
const removeAbortListeners = () => {
|
|
879
|
+
for (const signal of abortSignals) signal.removeEventListener("abort", handleAbort);
|
|
880
|
+
};
|
|
881
|
+
const timeout = setTimeout(() => {
|
|
882
|
+
removeAbortListeners();
|
|
883
|
+
resolve();
|
|
884
|
+
}, ms);
|
|
885
|
+
function handleAbort(event) {
|
|
886
|
+
clearTimeout(timeout);
|
|
887
|
+
removeAbortListeners();
|
|
888
|
+
reject(getAbortReason(event.currentTarget));
|
|
889
|
+
}
|
|
890
|
+
for (const signal of abortSignals) signal.addEventListener("abort", handleAbort, { once: true });
|
|
891
|
+
});
|
|
892
|
+
}
|
|
893
|
+
function getAbortedSignal(...signals) {
|
|
894
|
+
return signals.find((signal) => signal?.aborted);
|
|
895
|
+
}
|
|
896
|
+
function getAbortReason(signal) {
|
|
897
|
+
return signal.reason ?? new DOMException("The operation was aborted.", "AbortError");
|
|
898
|
+
}
|
|
845
899
|
/**
|
|
846
900
|
* Performs a double-knock request to the given URL. For the details of
|
|
847
901
|
* double-knocking, see
|
|
@@ -868,10 +922,7 @@ async function doubleKnockInternal(request, identity, options, redirected = 0, v
|
|
|
868
922
|
body
|
|
869
923
|
});
|
|
870
924
|
log?.(signedRequest);
|
|
871
|
-
let response = await
|
|
872
|
-
redirect: "manual",
|
|
873
|
-
signal
|
|
874
|
-
});
|
|
925
|
+
let response = await fetchDoubleKnockRequest(request, signedRequest, signal);
|
|
875
926
|
if (response.status >= 300 && response.status < 400 && response.headers.has("Location")) {
|
|
876
927
|
if (redirected >= maximumRedirection) throw new FetchError(request.url, `Too many redirections (${redirected + 1})`);
|
|
877
928
|
const redirectRequest = createRedirectRequest(request, response.headers.get("Location"), body);
|
|
@@ -946,10 +997,7 @@ async function doubleKnockInternal(request, identity, options, redirected = 0, v
|
|
|
946
997
|
body
|
|
947
998
|
});
|
|
948
999
|
log?.(signedRequest);
|
|
949
|
-
response = await
|
|
950
|
-
redirect: "manual",
|
|
951
|
-
signal
|
|
952
|
-
});
|
|
1000
|
+
response = await fetchDoubleKnockRequest(request, signedRequest, signal);
|
|
953
1001
|
if (response.status >= 300 && response.status < 400 && response.headers.has("Location")) {
|
|
954
1002
|
if (redirected >= maximumRedirection) throw new FetchError(request.url, `Too many redirections (${redirected + 1})`);
|
|
955
1003
|
const redirectRequest = createRedirectRequest(request, response.headers.get("Location"), body);
|
|
@@ -1,6 +1,7 @@
|
|
|
1
|
+
/// <reference lib="esnext.temporal" />
|
|
1
2
|
import { CryptographicKey, Multikey } from "@fedify/vocab";
|
|
2
|
-
import { DocumentLoader } from "@fedify/vocab-runtime";
|
|
3
3
|
import { TracerProvider } from "@opentelemetry/api";
|
|
4
|
+
import { DocumentLoader } from "@fedify/vocab-runtime";
|
|
4
5
|
|
|
5
6
|
//#region src/sig/key.d.ts
|
|
6
7
|
/**
|