@fedify/fedify 2.3.0-dev.1079 → 2.3.0-dev.1110

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.
Files changed (86) hide show
  1. package/dist/{builder-DqT-OPZ8.mjs → builder-B-Y6fwSu.mjs} +2 -2
  2. package/dist/{chunk-nlSIicah.js → chunk-CRNNMoPX.js} +2 -2
  3. package/dist/{client-z-8dc-e1.d.cts → client-CAM_bQXx.d.cts} +1 -0
  4. package/dist/{client-AtlibPOU.d.ts → client-CSddvgWN.d.ts} +1 -2
  5. package/dist/compat/mod.d.cts +2 -1
  6. package/dist/compat/mod.d.ts +2 -3
  7. package/dist/compat/mod.js +3 -3
  8. package/dist/compat/transformers.test.mjs +1 -1
  9. package/dist/{context-DrNqYkPw.d.ts → context-C0C_sRha.d.cts} +6 -7
  10. package/dist/{context-BKLGj9QO.d.cts → context-Dqgt8saU.d.ts} +6 -5
  11. package/dist/{deno-BxqkYxIf.mjs → deno-hqC7tKJn.mjs} +1 -1
  12. package/dist/{docloader-vsZP01ww.mjs → docloader-BOEuuXkX.mjs} +2 -2
  13. package/dist/federation/builder.test.mjs +1 -1
  14. package/dist/federation/handler.test.mjs +84 -2
  15. package/dist/federation/idempotency.test.mjs +2 -2
  16. package/dist/federation/middleware.test.mjs +383 -16
  17. package/dist/federation/mod.cjs +1 -1
  18. package/dist/federation/mod.d.cts +4 -3
  19. package/dist/federation/mod.d.ts +4 -5
  20. package/dist/federation/mod.js +2 -2
  21. package/dist/federation/send.test.mjs +3 -3
  22. package/dist/federation/webfinger.test.mjs +1 -1
  23. package/dist/{http-CrGuipxe.d.cts → http-BDZeS5om.d.ts} +2 -1
  24. package/dist/{http-DM4ZAFCe.mjs → http-BLopFpvC.mjs} +58 -10
  25. package/dist/{http-aQzN9Ayi.d.ts → http-C87EWkO0.d.cts} +2 -3
  26. package/dist/{http-DQ25_ruv.cjs → http-DV0il3vk.cjs} +57 -9
  27. package/dist/{http-b__OS_rJ.js → http-O8MYWwk8.js} +58 -10
  28. package/dist/{key-BNp_b9oE.mjs → key-DW1EVmtP.mjs} +1 -1
  29. package/dist/{kv-CbLNp3zQ.d.cts → kv-D6hNiMTK.d.ts} +1 -0
  30. package/dist/{kv-cache-D5vjOi5y.js → kv-cache-C3NWWiTg.js} +2 -2
  31. package/dist/{kv-cache-EaVpV2xQ.mjs → kv-cache-CiiNwT6W.mjs} +1 -1
  32. package/dist/{kv-cache-DzctboPD.cjs → kv-cache-Dya-TWMe.cjs} +2 -2
  33. package/dist/{kv-GFYnFoOl.d.ts → kv-gJ8LYbxX.d.cts} +1 -3
  34. package/dist/{ld-L9w529xq.mjs → ld-BNkk2Yal.mjs} +2 -2
  35. package/dist/{middleware-DZNHpEbh.js → middleware-CjzI3aYo.js} +309 -111
  36. package/dist/{middleware-CpCSD43m.mjs → middleware-D6FbOjuK.mjs} +1 -1
  37. package/dist/{middleware-CTdNwf_s.mjs → middleware-DA2WTBr4.mjs} +188 -114
  38. package/dist/{middleware-CMF242Rg.cjs → middleware-DUWeXjZR.cjs} +305 -107
  39. package/dist/{mod-CLgIXe9w.d.ts → mod-B0rWmfW5.d.cts} +4 -5
  40. package/dist/{mod-CMEbIaNh.d.cts → mod-BhU_H1I_.d.ts} +4 -3
  41. package/dist/{mod-B8Z8mBLk.d.ts → mod-CLPnQPsv.d.cts} +2 -3
  42. package/dist/{mod-Cr3f-ACa.d.cts → mod-DHO9lk3D.d.ts} +3 -2
  43. package/dist/{mod-CR8soWa9.d.ts → mod-DXY9JF28.d.cts} +3 -4
  44. package/dist/{mod-DClCOv0M.d.cts → mod-Dx3-hqyo.d.ts} +2 -1
  45. package/dist/mod.cjs +4 -4
  46. package/dist/mod.d.cts +9 -8
  47. package/dist/mod.d.ts +9 -10
  48. package/dist/mod.js +9 -9
  49. package/dist/nodeinfo/handler.test.mjs +1 -1
  50. package/dist/nodeinfo/mod.d.cts +2 -1
  51. package/dist/nodeinfo/mod.d.ts +2 -3
  52. package/dist/nodeinfo/mod.js +3 -3
  53. package/dist/otel/mod.d.cts +2 -1
  54. package/dist/otel/mod.d.ts +2 -3
  55. package/dist/otel/mod.js +2 -2
  56. package/dist/{owner-CptqhsOy.d.cts → owner-CnngXDNJ.d.ts} +2 -1
  57. package/dist/{owner-74ARJ5TL.d.ts → owner-DEvZuyOE.d.cts} +2 -3
  58. package/dist/{owner-C30LGgMz.mjs → owner-jvJAtR5O.mjs} +2 -2
  59. package/dist/{proof-CO1qAbN8.js → proof-5kT7OUPV.js} +2 -2
  60. package/dist/{proof-zrLeLWgt.cjs → proof-BD92WeqV.cjs} +1 -1
  61. package/dist/{proof-BjToRsXF.mjs → proof-mfmHH9j0.mjs} +2 -2
  62. package/dist/runtime/mod.d.cts +1 -0
  63. package/dist/runtime/mod.d.ts +1 -2
  64. package/dist/runtime/mod.js +3 -3
  65. package/dist/{send-DzbMznU6.mjs → send-hokVCPu6.mjs} +127 -3
  66. package/dist/sig/http.test.mjs +120 -2
  67. package/dist/sig/key.test.mjs +1 -1
  68. package/dist/sig/ld.test.mjs +2 -2
  69. package/dist/sig/mod.cjs +2 -2
  70. package/dist/sig/mod.d.cts +4 -3
  71. package/dist/sig/mod.d.ts +4 -5
  72. package/dist/sig/mod.js +4 -4
  73. package/dist/sig/owner.test.mjs +1 -1
  74. package/dist/sig/proof.test.mjs +1 -1
  75. package/dist/{transformers-ve6e2xcg.js → transformers-BGMIq1cs.js} +2 -2
  76. package/dist/{types-hvL8ElAs.js → types-CAY3OdLq.js} +2 -2
  77. package/dist/utils/docloader.test.mjs +2 -2
  78. package/dist/utils/kv-cache.test.mjs +1 -1
  79. package/dist/utils/mod.cjs +1 -1
  80. package/dist/utils/mod.d.cts +2 -1
  81. package/dist/utils/mod.d.ts +2 -3
  82. package/dist/utils/mod.js +3 -3
  83. package/dist/vocab/mod.d.cts +1 -0
  84. package/dist/vocab/mod.d.ts +1 -2
  85. package/dist/vocab/mod.js +2 -2
  86. package/package.json +7 -7
@@ -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-DM4ZAFCe.mjs";
14
+ import { l as verifyRequest, s as signRequest } from "../http-BLopFpvC.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-vsZP01ww.mjs";
17
- import { a as signJsonLd, o as verifyJsonLd, r as detachSignature } from "../ld-L9w529xq.mjs";
18
- import { t as doesActorOwnKey } from "../owner-C30LGgMz.mjs";
19
- import { i as verifyObject, r as signObject } from "../proof-BjToRsXF.mjs";
16
+ import { t as getAuthenticatedDocumentLoader } from "../docloader-BOEuuXkX.mjs";
17
+ import { a as signJsonLd, o as verifyJsonLd, r as detachSignature } from "../ld-BNkk2Yal.mjs";
18
+ import { t as doesActorOwnKey } from "../owner-jvJAtR5O.mjs";
19
+ import { i as verifyObject, r as signObject } from "../proof-mfmHH9j0.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-CTdNwf_s.mjs";
21
+ import { i as KvSpecDeterminer, n as FederationImpl, o as createFederation, r as InboxContextImpl, t as ContextImpl } from "../middleware-DA2WTBr4.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
- queue: {
1925
- enqueue(message) {
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";
@@ -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-CMF242Rg.cjs");
5
+ const require_middleware = require("../middleware-DUWeXjZR.cjs");
6
6
  let es_toolkit = require("es-toolkit");
7
7
  //#region src/federation/kv.ts
8
8
  /**
@@ -1,4 +1,5 @@
1
- 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-BKLGj9QO.cjs";
2
- import { a as MemoryKvStore, i as KvStoreSetOptions, n as KvStore, r as KvStoreListEntry, t as KvKey } from "../kv-CbLNp3zQ.cjs";
3
- import { n as handleWebFinger, t as WebFingerHandlerParameters } from "../mod-DClCOv0M.cjs";
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 };
@@ -1,6 +1,5 @@
1
- import { Temporal } from "@js-temporal/polyfill";
2
- import { URLPattern } from "urlpattern-polyfill";
3
- 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-DrNqYkPw.js";
4
- import { a as MemoryKvStore, i as KvStoreSetOptions, n as KvStore, r as KvStoreListEntry, t as KvKey } from "../kv-GFYnFoOl.js";
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 };
@@ -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-DZNHpEbh.js";
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-CjzI3aYo.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-DM4ZAFCe.mjs";
12
+ import { l as verifyRequest } from "../http-BLopFpvC.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-C30LGgMz.mjs";
15
- import { n as extractInboxes, r as sendActivity, t as SendActivityError } from "../send-DzbMznU6.mjs";
14
+ import { t as doesActorOwnKey } from "../owner-jvJAtR5O.mjs";
15
+ import { n as extractInboxes, r as sendActivity, t as SendActivityError } from "../send-hokVCPu6.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-CTdNwf_s.mjs";
8
+ import { o as createFederation, s as handleWebFinger } from "../middleware-DA2WTBr4.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,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
  /**
@@ -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-BxqkYxIf.mjs";
4
+ import { n as version, t as name } from "./deno-hqC7tKJn.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-BNp_b9oE.mjs";
6
+ import { o as validateCryptoKey, r as fetchKeyDetailed } from "./key-DW1EVmtP.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 fetch(signedRequest, {
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 fetch(signedRequest, {
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);