@fedify/fedify 1.4.6 → 1.4.8

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 (49) hide show
  1. package/CHANGES.md +140 -1
  2. package/esm/deno.js +1 -1
  3. package/esm/deps/jsr.io/@std/encoding/1.0.8/_common16.js +45 -0
  4. package/esm/deps/jsr.io/@std/encoding/1.0.8/_common64.js +98 -0
  5. package/esm/deps/jsr.io/@std/encoding/1.0.8/_common_detach.js +13 -0
  6. package/esm/deps/jsr.io/@std/encoding/1.0.8/base64.js +82 -0
  7. package/esm/deps/jsr.io/@std/encoding/1.0.8/base64url.js +72 -0
  8. package/esm/deps/jsr.io/@std/encoding/1.0.8/hex.js +87 -0
  9. package/esm/federation/collection.js +3 -3
  10. package/esm/federation/handler.js +5 -2
  11. package/esm/federation/middleware.js +20 -3
  12. package/esm/runtime/key.js +3 -3
  13. package/esm/sig/http.js +2 -2
  14. package/esm/sig/ld.js +2 -2
  15. package/esm/sig/proof.js +1 -1
  16. package/esm/vocab/lookup.js +1 -1
  17. package/esm/vocab/vocab.js +176 -176
  18. package/esm/webfinger/handler.js +12 -5
  19. package/package.json +1 -1
  20. package/types/deps/jsr.io/@std/async/{1.0.11 → 1.0.12}/delay.d.ts.map +1 -1
  21. package/types/deps/jsr.io/@std/encoding/1.0.8/_common16.d.ts +21 -0
  22. package/types/deps/jsr.io/@std/encoding/1.0.8/_common16.d.ts.map +1 -0
  23. package/types/deps/jsr.io/@std/encoding/1.0.8/_common64.d.ts +21 -0
  24. package/types/deps/jsr.io/@std/encoding/1.0.8/_common64.d.ts.map +1 -0
  25. package/types/deps/jsr.io/@std/encoding/1.0.8/_common_detach.d.ts +4 -0
  26. package/types/deps/jsr.io/@std/encoding/1.0.8/_common_detach.d.ts.map +1 -0
  27. package/types/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/_types.d.ts.map +1 -1
  28. package/types/deps/jsr.io/@std/encoding/1.0.8/base64.d.ts.map +1 -0
  29. package/types/deps/jsr.io/@std/encoding/1.0.8/base64url.d.ts.map +1 -0
  30. package/types/deps/jsr.io/@std/encoding/1.0.8/hex.d.ts.map +1 -0
  31. package/types/federation/collection.d.ts +2 -2
  32. package/types/federation/handler.d.ts.map +1 -1
  33. package/types/federation/middleware.d.ts.map +1 -1
  34. package/esm/deps/jsr.io/@std/encoding/1.0.7/_validate_binary_like.js +0 -26
  35. package/esm/deps/jsr.io/@std/encoding/1.0.7/base64.js +0 -163
  36. package/esm/deps/jsr.io/@std/encoding/1.0.7/base64url.js +0 -81
  37. package/esm/deps/jsr.io/@std/encoding/1.0.7/hex.js +0 -109
  38. package/types/deps/jsr.io/@std/encoding/1.0.7/_validate_binary_like.d.ts +0 -2
  39. package/types/deps/jsr.io/@std/encoding/1.0.7/_validate_binary_like.d.ts.map +0 -1
  40. package/types/deps/jsr.io/@std/encoding/1.0.7/base64.d.ts.map +0 -1
  41. package/types/deps/jsr.io/@std/encoding/1.0.7/base64url.d.ts.map +0 -1
  42. package/types/deps/jsr.io/@std/encoding/1.0.7/hex.d.ts.map +0 -1
  43. /package/esm/deps/jsr.io/@std/async/{1.0.11 → 1.0.12}/delay.js +0 -0
  44. /package/esm/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/_types.js +0 -0
  45. /package/types/deps/jsr.io/@std/async/{1.0.11 → 1.0.12}/delay.d.ts +0 -0
  46. /package/types/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/_types.d.ts +0 -0
  47. /package/types/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/base64.d.ts +0 -0
  48. /package/types/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/base64url.d.ts +0 -0
  49. /package/types/deps/jsr.io/@std/encoding/{1.0.7 → 1.0.8}/hex.d.ts +0 -0
package/CHANGES.md CHANGED
@@ -3,6 +3,33 @@
3
3
  Fedify changelog
4
4
  ================
5
5
 
6
+ Version 1.4.8
7
+ -------------
8
+
9
+ Released on March 26, 2025.
10
+
11
+ - Fixed a bug where the `totalItems` property of `OrderedCollection`
12
+ objects returned by followers collection dispatcher had been an incorrect
13
+ value if a `base-url` parameter was provided.
14
+
15
+ - Fixed a bug where the `id` property of `OrderedCollection` and
16
+ `OrderedCollectionPage` objects returned by followers collection dispatcher
17
+ had been an incorrect value if a `base-url` parameter was provided.
18
+
19
+
20
+ Version 1.4.7
21
+ -------------
22
+
23
+ Released on March 20, 2025.
24
+
25
+ - Fixed a bug of WebFinger handler where it had failed to match
26
+ `acct:` URIs with a host having a port number.
27
+ [[#218], [#219] by Revath S Kumar]
28
+
29
+ - Fixed a server error thrown when an invalid URL was passed to the `base-url`
30
+ parameter of the followers collection. [[#217]]
31
+
32
+
6
33
  Version 1.4.6
7
34
  -------------
8
35
 
@@ -166,6 +193,33 @@ Released on February 5, 2025.
166
193
  [#195]: https://github.com/fedify-dev/fedify/issues/195
167
194
 
168
195
 
196
+ Version 1.3.15
197
+ --------------
198
+
199
+ Released on March 26, 2025.
200
+
201
+ - Fixed a bug where the `totalItems` property of `OrderedCollection`
202
+ objects returned by followers collection dispatcher had been an incorrect
203
+ value if a `base-url` parameter was provided.
204
+
205
+ - Fixed a bug where the `id` property of `OrderedCollection` and
206
+ `OrderedCollectionPage` objects returned by followers collection dispatcher
207
+ had been an incorrect value if a `base-url` parameter was provided.
208
+
209
+
210
+ Version 1.3.14
211
+ --------------
212
+
213
+ Released on March 20, 2025.
214
+
215
+ - Fixed a bug of WebFinger handler where it had failed to match
216
+ `acct:` URIs with a host having a port number.
217
+ [[#218], [#219] by Revath S Kumar]
218
+
219
+ - Fixed a server error thrown when an invalid URL was passed to the `base-url`
220
+ parameter of the followers collection. [[#217]]
221
+
222
+
169
223
  Version 1.3.13
170
224
  --------------
171
225
 
@@ -445,6 +499,33 @@ Released on November 30, 2024.
445
499
  [#193]: https://github.com/fedify-dev/fedify/issues/193
446
500
 
447
501
 
502
+ Version 1.2.19
503
+ --------------
504
+
505
+ Released on March 26, 2025.
506
+
507
+ - Fixed a bug where the `totalItems` property of `OrderedCollection`
508
+ objects returned by followers collection dispatcher had been an incorrect
509
+ value if a `base-url` parameter was provided.
510
+
511
+ - Fixed a bug where the `id` property of `OrderedCollection` and
512
+ `OrderedCollectionPage` objects returned by followers collection dispatcher
513
+ had been an incorrect value if a `base-url` parameter was provided.
514
+
515
+
516
+ Version 1.2.18
517
+ --------------
518
+
519
+ Released on March 20, 2025.
520
+
521
+ - Fixed a bug of WebFinger handler where it had failed to match
522
+ `acct:` URIs with a host having a port number.
523
+ [[#218], [#219] by Revath S Kumar]
524
+
525
+ - Fixed a server error thrown when an invalid URL was passed to the `base-url`
526
+ parameter of the followers collection. [[#217]]
527
+
528
+
448
529
  Version 1.2.17
449
530
  --------------
450
531
 
@@ -773,6 +854,33 @@ Released on October 31, 2024.
773
854
  [#118]: https://github.com/fedify-dev/fedify/issues/118
774
855
 
775
856
 
857
+ Version 1.1.19
858
+ --------------
859
+
860
+ Released on March 26, 2025.
861
+
862
+ - Fixed a bug where the `totalItems` property of `OrderedCollection`
863
+ objects returned by followers collection dispatcher had been an incorrect
864
+ value if a `base-url` parameter was provided.
865
+
866
+ - Fixed a bug where the `id` property of `OrderedCollection` and
867
+ `OrderedCollectionPage` objects returned by followers collection dispatcher
868
+ had been an incorrect value if a `base-url` parameter was provided.
869
+
870
+
871
+ Version 1.1.18
872
+ --------------
873
+
874
+ Released on March 20, 2025.
875
+
876
+ - Fixed a bug of WebFinger handler where it had failed to match
877
+ `acct:` URIs with a host having a port number.
878
+ [[#218], [#219] by Revath S Kumar]
879
+
880
+ - Fixed a server error thrown when an invalid URL was passed to the `base-url`
881
+ parameter of the followers collection. [[#217]]
882
+
883
+
776
884
  Version 1.1.17
777
885
  --------------
778
886
 
@@ -1142,6 +1250,37 @@ Released on October 20, 2024.
1142
1250
  [#150]: https://github.com/fedify-dev/fedify/issues/150
1143
1251
 
1144
1252
 
1253
+ Version 1.0.22
1254
+ --------------
1255
+
1256
+ Released on March 26, 2025.
1257
+
1258
+ - Fixed a bug where the `totalItems` property of `OrderedCollection`
1259
+ objects returned by followers collection dispatcher had been an incorrect
1260
+ value if a `base-url` parameter was provided.
1261
+
1262
+ - Fixed a bug where the `id` property of `OrderedCollection` and
1263
+ `OrderedCollectionPage` objects returned by followers collection dispatcher
1264
+ had been an incorrect value if a `base-url` parameter was provided.
1265
+
1266
+
1267
+ Version 1.0.21
1268
+ --------------
1269
+
1270
+ Released on March 20, 2025.
1271
+
1272
+ - Fixed a bug of WebFinger handler where it had failed to match
1273
+ `acct:` URIs with a host having a port number.
1274
+ [[#218], [#219] by Revath S Kumar]
1275
+
1276
+ - Fixed a server error thrown when an invalid URL was passed to the `base-url`
1277
+ parameter of the followers collection. [[#217]]
1278
+
1279
+ [#217]: https://github.com/fedify-dev/fedify/issues/217
1280
+ [#218]: https://github.com/fedify-dev/fedify/issues/218
1281
+ [#219]: https://github.com/fedify-dev/fedify/pull/219
1282
+
1283
+
1145
1284
  Version 1.0.20
1146
1285
  --------------
1147
1286
 
@@ -3307,4 +3446,4 @@ Version 0.1.0
3307
3446
  Initial release. Released on March 8, 2024.
3308
3447
 
3309
3448
  <!-- cSpell: ignore Dogeon Fabien Wressell Emelia Fróði Karlsson -->
3310
- <!-- cSpell: ignore Hana Heesun Kyunghee Jiyu -->
3449
+ <!-- cSpell: ignore Hana Heesun Kyunghee Jiyu Revath Kumar -->
package/esm/deno.js CHANGED
@@ -1,6 +1,6 @@
1
1
  export default {
2
2
  "name": "@fedify/fedify",
3
- "version": "1.4.6",
3
+ "version": "1.4.8",
4
4
  "license": "MIT",
5
5
  "exports": {
6
6
  ".": "./mod.ts",
@@ -0,0 +1,45 @@
1
+ // Copyright 2018-2025 the Deno authors. MIT license.
2
+ /**
3
+ * Calculate the output size needed to encode a given input size for
4
+ * {@linkcode encodeRawHex}.
5
+ *
6
+ * @param originalSize The size of the input buffer.
7
+ * @returns The size of the output buffer.
8
+ *
9
+ * @example Basic Usage
10
+ * ```ts
11
+ * import { assertEquals } from "@std/assert";
12
+ * import { calcMax } from "@std/encoding/unstable-hex";
13
+ *
14
+ * assertEquals(calcMax(1), 2);
15
+ * ```
16
+ */
17
+ export function calcMax(originalSize) {
18
+ return originalSize * 2;
19
+ }
20
+ export function encode(buffer, i, o, alphabet) {
21
+ for (; i < buffer.length; ++i) {
22
+ const x = buffer[i];
23
+ buffer[o++] = alphabet[x >> 4];
24
+ buffer[o++] = alphabet[x & 0xF];
25
+ }
26
+ return o;
27
+ }
28
+ export function decode(buffer, i, o, alphabet) {
29
+ if ((buffer.length - o) % 2 === 1) {
30
+ throw new RangeError(`Cannot decode input as hex: Length (${buffer.length - o}) must be divisible by 2`);
31
+ }
32
+ i += 1;
33
+ for (; i < buffer.length; i += 2) {
34
+ buffer[o++] = (getByte(buffer[i - 1], alphabet) << 4) |
35
+ getByte(buffer[i], alphabet);
36
+ }
37
+ return o;
38
+ }
39
+ function getByte(char, alphabet) {
40
+ const byte = alphabet[char] ?? 16;
41
+ if (byte === 16) { // alphabet.Hex.length
42
+ throw new TypeError(`Cannot decode input as hex: Invalid character (${String.fromCharCode(char)})`);
43
+ }
44
+ return byte;
45
+ }
@@ -0,0 +1,98 @@
1
+ // Copyright 2018-2025 the Deno authors. MIT license.
2
+ /**
3
+ * Calculate the output size needed to encode a given input size for
4
+ * {@linkcode encodeRawBase64}.
5
+ *
6
+ * @param originalSize The size of the input buffer.
7
+ * @returns The size of the output buffer.
8
+ *
9
+ * @example Basic Usage
10
+ * ```ts
11
+ * import { assertEquals } from "@std/assert";
12
+ * import { calcMax } from "@std/encoding/unstable-base64";
13
+ *
14
+ * assertEquals(calcMax(1), 4);
15
+ * ```
16
+ */
17
+ export function calcMax(originalSize) {
18
+ return ((originalSize + 2) / 3 | 0) * 4;
19
+ }
20
+ export function encode(buffer, i, o, alphabet, padding) {
21
+ i += 2;
22
+ for (; i < buffer.length; i += 3) {
23
+ const x = (buffer[i - 2] << 16) | (buffer[i - 1] << 8) | buffer[i];
24
+ buffer[o++] = alphabet[x >> 18];
25
+ buffer[o++] = alphabet[x >> 12 & 0x3F];
26
+ buffer[o++] = alphabet[x >> 6 & 0x3F];
27
+ buffer[o++] = alphabet[x & 0x3F];
28
+ }
29
+ switch (i) {
30
+ case buffer.length + 1: {
31
+ const x = buffer[i - 2] << 16;
32
+ buffer[o++] = alphabet[x >> 18];
33
+ buffer[o++] = alphabet[x >> 12 & 0x3F];
34
+ buffer[o++] = padding;
35
+ buffer[o++] = padding;
36
+ break;
37
+ }
38
+ case buffer.length: {
39
+ const x = (buffer[i - 2] << 16) | (buffer[i - 1] << 8);
40
+ buffer[o++] = alphabet[x >> 18];
41
+ buffer[o++] = alphabet[x >> 12 & 0x3F];
42
+ buffer[o++] = alphabet[x >> 6 & 0x3F];
43
+ buffer[o++] = padding;
44
+ break;
45
+ }
46
+ }
47
+ return o;
48
+ }
49
+ export function decode(buffer, i, o, alphabet, padding) {
50
+ for (let x = buffer.length - 2; x < buffer.length; ++x) {
51
+ if (buffer[x] === padding) {
52
+ for (let y = x + 1; y < buffer.length; ++y) {
53
+ if (buffer[y] !== padding) {
54
+ throw new TypeError(`Cannot decode input as base64: Invalid character (${String.fromCharCode(buffer[y])})`);
55
+ }
56
+ }
57
+ buffer = buffer.subarray(0, x);
58
+ break;
59
+ }
60
+ }
61
+ if ((buffer.length - o) % 4 === 1) {
62
+ throw new RangeError(`Cannot decode input as base64: Length (${buffer.length - o}), excluding padding, must not have a remainder of 1 when divided by 4`);
63
+ }
64
+ i += 3;
65
+ for (; i < buffer.length; i += 4) {
66
+ const x = (getByte(buffer[i - 3], alphabet) << 18) |
67
+ (getByte(buffer[i - 2], alphabet) << 12) |
68
+ (getByte(buffer[i - 1], alphabet) << 6) |
69
+ getByte(buffer[i], alphabet);
70
+ buffer[o++] = x >> 16;
71
+ buffer[o++] = x >> 8 & 0xFF;
72
+ buffer[o++] = x & 0xFF;
73
+ }
74
+ switch (i) {
75
+ case buffer.length + 1: {
76
+ const x = (getByte(buffer[i - 3], alphabet) << 18) |
77
+ (getByte(buffer[i - 2], alphabet) << 12);
78
+ buffer[o++] = x >> 16;
79
+ break;
80
+ }
81
+ case buffer.length: {
82
+ const x = (getByte(buffer[i - 3], alphabet) << 18) |
83
+ (getByte(buffer[i - 2], alphabet) << 12) |
84
+ (getByte(buffer[i - 1], alphabet) << 6);
85
+ buffer[o++] = x >> 16;
86
+ buffer[o++] = x >> 8 & 0xFF;
87
+ break;
88
+ }
89
+ }
90
+ return o;
91
+ }
92
+ function getByte(char, alphabet) {
93
+ const byte = alphabet[char] ?? 64;
94
+ if (byte === 64) { // alphabet.Base64.length
95
+ throw new TypeError(`Cannot decode input as base64: Invalid character (${String.fromCharCode(char)})`);
96
+ }
97
+ return byte;
98
+ }
@@ -0,0 +1,13 @@
1
+ // Copyright 2018-2025 the Deno authors. MIT license.
2
+ export function detach(buffer, maxSize) {
3
+ const originalSize = buffer.length;
4
+ if (buffer.byteOffset) {
5
+ const b = new Uint8Array(buffer.buffer);
6
+ b.set(buffer);
7
+ buffer = b.subarray(0, originalSize);
8
+ }
9
+ // deno-lint-ignore no-explicit-any
10
+ buffer = new Uint8Array(buffer.buffer.transfer(maxSize));
11
+ buffer.set(buffer.subarray(0, originalSize), maxSize - originalSize);
12
+ return [buffer, maxSize - originalSize];
13
+ }
@@ -0,0 +1,82 @@
1
+ // Copyright 2018-2025 the Deno authors. MIT license.
2
+ // This module is browser compatible.
3
+ /**
4
+ * Utilities for
5
+ * {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-4 | base64}
6
+ * encoding and decoding.
7
+ *
8
+ * ```ts
9
+ * import {
10
+ * encodeBase64,
11
+ * decodeBase64,
12
+ * } from "@std/encoding/base64";
13
+ * import { assertEquals } from "@std/assert";
14
+ *
15
+ * const foobar = new TextEncoder().encode("foobar");
16
+ *
17
+ * assertEquals(encodeBase64(foobar), "Zm9vYmFy");
18
+ * assertEquals(decodeBase64("Zm9vYmFy"), foobar);
19
+ * ```
20
+ *
21
+ * @module
22
+ */
23
+ import { calcMax, decode, encode } from "./_common64.js";
24
+ import { detach } from "./_common_detach.js";
25
+ const padding = "=".charCodeAt(0);
26
+ const alphabet = new TextEncoder()
27
+ .encode("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/");
28
+ const rAlphabet = new Uint8Array(128).fill(64); // alphabet.length
29
+ alphabet.forEach((byte, i) => rAlphabet[byte] = i);
30
+ /**
31
+ * Converts data into a base64-encoded string.
32
+ *
33
+ * @see {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-4}
34
+ *
35
+ * @param data The data to encode.
36
+ * @returns The base64-encoded string.
37
+ *
38
+ * @example Usage
39
+ * ```ts
40
+ * import { encodeBase64 } from "@std/encoding/base64";
41
+ * import { assertEquals } from "@std/assert";
42
+ *
43
+ * assertEquals(encodeBase64("foobar"), "Zm9vYmFy");
44
+ * ```
45
+ */
46
+ export function encodeBase64(data) {
47
+ if (typeof data === "string") {
48
+ data = new TextEncoder().encode(data);
49
+ }
50
+ else if (data instanceof ArrayBuffer)
51
+ data = new Uint8Array(data).slice();
52
+ else
53
+ data = data.slice();
54
+ const [output, i] = detach(data, calcMax(data.length));
55
+ encode(output, i, 0, alphabet, padding);
56
+ return new TextDecoder().decode(output);
57
+ }
58
+ /**
59
+ * Decodes a base64-encoded string.
60
+ *
61
+ * @see {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-4}
62
+ *
63
+ * @param b64 The base64-encoded string to decode.
64
+ * @returns The decoded data.
65
+ *
66
+ * @example Usage
67
+ * ```ts
68
+ * import { decodeBase64 } from "@std/encoding/base64";
69
+ * import { assertEquals } from "@std/assert";
70
+ *
71
+ * assertEquals(
72
+ * decodeBase64("Zm9vYmFy"),
73
+ * new TextEncoder().encode("foobar")
74
+ * );
75
+ * ```
76
+ */
77
+ export function decodeBase64(b64) {
78
+ const output = new TextEncoder().encode(b64);
79
+ // deno-lint-ignore no-explicit-any
80
+ return new Uint8Array(output.buffer
81
+ .transfer(decode(output, 0, 0, rAlphabet, padding)));
82
+ }
@@ -0,0 +1,72 @@
1
+ // Copyright 2018-2025 the Deno authors. MIT license.
2
+ // This module is browser compatible.
3
+ /**
4
+ * Utilities for
5
+ * {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-5 | base64url}
6
+ * encoding and decoding.
7
+ *
8
+ * @module
9
+ */
10
+ import { calcMax, decode, encode } from "./_common64.js";
11
+ import { detach } from "./_common_detach.js";
12
+ const padding = "=".charCodeAt(0);
13
+ const alphabet = new TextEncoder()
14
+ .encode("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_");
15
+ const rAlphabet = new Uint8Array(128).fill(64); // alphabet.length
16
+ alphabet.forEach((byte, i) => rAlphabet[byte] = i);
17
+ /**
18
+ * Convert data into a base64url-encoded string.
19
+ *
20
+ * @see {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-5}
21
+ *
22
+ * @param data The data to encode.
23
+ * @returns The base64url-encoded string.
24
+ *
25
+ * @example Usage
26
+ * ```ts
27
+ * import { encodeBase64Url } from "@std/encoding/base64url";
28
+ * import { assertEquals } from "@std/assert";
29
+ *
30
+ * assertEquals(encodeBase64Url("foobar"), "Zm9vYmFy");
31
+ * ```
32
+ */
33
+ export function encodeBase64Url(data) {
34
+ if (typeof data === "string") {
35
+ data = new TextEncoder().encode(data);
36
+ }
37
+ else if (data instanceof ArrayBuffer)
38
+ data = new Uint8Array(data).slice();
39
+ else
40
+ data = data.slice();
41
+ const [output, i] = detach(data, calcMax(data.length));
42
+ let o = encode(output, i, 0, alphabet, padding);
43
+ o = output.indexOf(padding, o - 2);
44
+ return new TextDecoder().decode(
45
+ // deno-lint-ignore no-explicit-any
46
+ o > 0 ? new Uint8Array(output.buffer.transfer(o)) : output);
47
+ }
48
+ /**
49
+ * Decodes a given base64url-encoded string.
50
+ *
51
+ * @see {@link https://www.rfc-editor.org/rfc/rfc4648.html#section-5}
52
+ *
53
+ * @param b64url The base64url-encoded string to decode.
54
+ * @returns The decoded data.
55
+ *
56
+ * @example Usage
57
+ * ```ts
58
+ * import { decodeBase64Url } from "@std/encoding/base64url";
59
+ * import { assertEquals } from "@std/assert";
60
+ *
61
+ * assertEquals(
62
+ * decodeBase64Url("Zm9vYmFy"),
63
+ * new TextEncoder().encode("foobar")
64
+ * );
65
+ * ```
66
+ */
67
+ export function decodeBase64Url(b64url) {
68
+ const output = new TextEncoder().encode(b64url);
69
+ // deno-lint-ignore no-explicit-any
70
+ return new Uint8Array(output.buffer
71
+ .transfer(decode(output, 0, 0, rAlphabet, padding)));
72
+ }
@@ -0,0 +1,87 @@
1
+ // Copyright 2009 The Go Authors. All rights reserved.
2
+ // https://github.com/golang/go/blob/master/LICENSE
3
+ // Copyright 2018-2025 the Deno authors. MIT license.
4
+ // This module is browser compatible.
5
+ /**
6
+ * Port of the Go
7
+ * {@link https://github.com/golang/go/blob/go1.12.5/src/encoding/hex/hex.go | encoding/hex}
8
+ * library.
9
+ *
10
+ * ```ts
11
+ * import {
12
+ * decodeHex,
13
+ * encodeHex,
14
+ * } from "@std/encoding/hex";
15
+ * import { assertEquals } from "@std/assert";
16
+ *
17
+ * assertEquals(encodeHex("abc"), "616263");
18
+ *
19
+ * assertEquals(
20
+ * decodeHex("616263"),
21
+ * new TextEncoder().encode("abc"),
22
+ * );
23
+ * ```
24
+ *
25
+ * @module
26
+ */
27
+ import { calcMax, decode, encode } from "./_common16.js";
28
+ import { detach } from "./_common_detach.js";
29
+ const alphabet = new TextEncoder()
30
+ .encode("0123456789abcdef");
31
+ const rAlphabet = new Uint8Array(128).fill(16); // alphabet.length
32
+ alphabet.forEach((byte, i) => rAlphabet[byte] = i);
33
+ new TextEncoder()
34
+ .encode("ABCDEF")
35
+ .forEach((byte, i) => rAlphabet[byte] = i + 10);
36
+ /**
37
+ * Converts data into a hex-encoded string.
38
+ *
39
+ * @param src The data to encode.
40
+ *
41
+ * @returns The hex-encoded string.
42
+ *
43
+ * @example Usage
44
+ * ```ts
45
+ * import { encodeHex } from "@std/encoding/hex";
46
+ * import { assertEquals } from "@std/assert";
47
+ *
48
+ * assertEquals(encodeHex("abc"), "616263");
49
+ * ```
50
+ */
51
+ export function encodeHex(src) {
52
+ if (typeof src === "string") {
53
+ src = new TextEncoder().encode(src);
54
+ }
55
+ else if (src instanceof ArrayBuffer)
56
+ src = new Uint8Array(src).slice();
57
+ else
58
+ src = src.slice();
59
+ const [output, i] = detach(src, calcMax(src.length));
60
+ encode(output, i, 0, alphabet);
61
+ return new TextDecoder().decode(output);
62
+ }
63
+ /**
64
+ * Decodes the given hex-encoded string. If the input is malformed, an error is
65
+ * thrown.
66
+ *
67
+ * @param src The hex-encoded string to decode.
68
+ *
69
+ * @returns The decoded data.
70
+ *
71
+ * @example Usage
72
+ * ```ts
73
+ * import { decodeHex } from "@std/encoding/hex";
74
+ * import { assertEquals } from "@std/assert";
75
+ *
76
+ * assertEquals(
77
+ * decodeHex("616263"),
78
+ * new TextEncoder().encode("abc"),
79
+ * );
80
+ * ```
81
+ */
82
+ export function decodeHex(src) {
83
+ const output = new TextEncoder().encode(src);
84
+ // deno-lint-ignore no-explicit-any
85
+ return new Uint8Array(output.buffer
86
+ .transfer(decode(output, 0, 0, rAlphabet)));
87
+ }
@@ -1,9 +1,9 @@
1
1
  import * as dntShim from "../_dnt.shims.js";
2
- import { encodeHex } from "../deps/jsr.io/@std/encoding/1.0.7/hex.js";
2
+ import { encodeHex } from "../deps/jsr.io/@std/encoding/1.0.8/hex.js";
3
3
  /**
4
4
  * Calculates the [partial follower collection digest][1].
5
5
  *
6
- * [1]: https://codeberg.org/fediverse/fep/src/branch/main/fep/8fcf/fep-8fcf.md#partial-follower-collection-digest
6
+ * [1]: https://w3id.org/fep/8fcf#partial-follower-collection-digest
7
7
  * @param uris The URIs to calculate the digest. Duplicate URIs are ignored.
8
8
  * @returns The digest.
9
9
  */
@@ -27,7 +27,7 @@ export async function digest(uris) {
27
27
  /**
28
28
  * Builds [`Collection-Synchronization`][1] header content.
29
29
  *
30
- * [1]: https://codeberg.org/fediverse/fep/src/branch/main/fep/8fcf/fep-8fcf.md#the-collection-synchronization-http-header
30
+ * [1]: https://w3id.org/fep/8fcf#the-collection-synchronization-http-header
31
31
  *
32
32
  * @param collectionId The sender's followers collection URI.
33
33
  * @param actorIds The actor URIs to digest.
@@ -84,7 +84,9 @@ export async function handleCollection(request, { name, identifier, uriGetter, f
84
84
  const baseUri = uriGetter(identifier);
85
85
  if (cursor == null) {
86
86
  const firstCursor = await collectionCallbacks.firstCursor?.(context, identifier);
87
- const totalItems = await collectionCallbacks.counter?.(context, identifier);
87
+ const totalItems = filter == null
88
+ ? await collectionCallbacks.counter?.(context, identifier)
89
+ : undefined;
88
90
  if (firstCursor == null) {
89
91
  const itemsOrResponse = await tracer.startActiveSpan(`activitypub.dispatch_collection ${spanName}`, {
90
92
  kind: SpanKind.SERVER,
@@ -223,7 +225,8 @@ function filterCollectionItems(items, collectionName, filterPredicate) {
223
225
  if (!logged) {
224
226
  getLogger(["fedify", "federation", "collection"]).warn(`The ${collectionName} collection apparently does not implement ` +
225
227
  "filtering. This may result in a large response payload. " +
226
- "Please consider implementing filtering for the collection.");
228
+ "Please consider implementing filtering for the collection. " +
229
+ "See also: https://fedify.dev/manual/collections#filtering-by-server");
227
230
  logged = true;
228
231
  }
229
232
  continue;
@@ -1440,13 +1440,24 @@ export class FederationImpl {
1440
1440
  case "followers": {
1441
1441
  let baseUrl = url.searchParams.get("base-url");
1442
1442
  if (baseUrl != null) {
1443
- const u = new URL(baseUrl);
1444
- baseUrl = `${u.origin}/`;
1443
+ try {
1444
+ baseUrl = `${new URL(baseUrl).origin}/`;
1445
+ }
1446
+ catch {
1447
+ // If base-url is invalid, set to null to behave as if it wasn't provided
1448
+ baseUrl = null;
1449
+ }
1445
1450
  }
1446
1451
  return await handleCollection(request, {
1447
1452
  name: "followers",
1448
1453
  identifier: route.values.identifier ?? route.values.handle,
1449
- uriGetter: context.getFollowersUri.bind(context),
1454
+ uriGetter: baseUrl == null
1455
+ ? context.getFollowersUri.bind(context)
1456
+ : (identifier) => {
1457
+ const uri = context.getFollowersUri(identifier);
1458
+ uri.searchParams.set("base-url", baseUrl);
1459
+ return uri;
1460
+ },
1450
1461
  context,
1451
1462
  filter: baseUrl != null ? new URL(baseUrl) : undefined,
1452
1463
  filterPredicate: baseUrl != null
@@ -2017,6 +2028,12 @@ export class ContextImpl {
2017
2028
  throw new Error("No first cursor dispatcher registered for followers collection.");
2018
2029
  }
2019
2030
  let cursor = await this.federation.followersCallbacks.firstCursor(this, identifier);
2031
+ if (cursor != null) {
2032
+ getLogger(["fedify", "federation", "outbox"]).warn("Since the followers collection dispatcher returned null for no " +
2033
+ "cursor (i.e., one-shot dispatcher), the pagination is used to fetch " +
2034
+ '"followers". However, it is recommended to implement the one-shot ' +
2035
+ "dispatcher for better performance.", { identifier });
2036
+ }
2020
2037
  while (cursor != null) {
2021
2038
  const result = await this.federation.followersCallbacks.dispatcher(this, identifier, cursor);
2022
2039
  if (result == null)
@@ -1,9 +1,9 @@
1
1
  import * as dntShim from "../_dnt.shims.js";
2
2
  import { createPublicKey } from "node:crypto";
3
3
  import { concat } from "../deps/jsr.io/@std/bytes/1.0.5/concat.js";
4
- import { decodeBase64, encodeBase64 } from "../deps/jsr.io/@std/encoding/1.0.7/base64.js";
5
- import { decodeBase64Url } from "../deps/jsr.io/@std/encoding/1.0.7/base64url.js";
6
- import { decodeHex } from "../deps/jsr.io/@std/encoding/1.0.7/hex.js";
4
+ import { decodeBase64, encodeBase64 } from "../deps/jsr.io/@std/encoding/1.0.8/base64.js";
5
+ import { decodeBase64Url } from "../deps/jsr.io/@std/encoding/1.0.8/base64url.js";
6
+ import { decodeHex } from "../deps/jsr.io/@std/encoding/1.0.8/hex.js";
7
7
  import { Integer, Sequence } from "asn1js";
8
8
  import { decode, encode } from "multibase";
9
9
  import { addPrefix, getCodeFromData, rmPrefix } from "multicodec";