Skip to content

Commit e047a11

Browse files
committed
lib: implement all 1-byte encodings in js
1 parent 2e597de commit e047a11

File tree

6 files changed

+192
-162
lines changed

6 files changed

+192
-162
lines changed

lib/internal/encoding.js

Lines changed: 38 additions & 32 deletions
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@ const {
1313
StringPrototypeSlice,
1414
Symbol,
1515
SymbolToStringTag,
16-
Uint8Array,
1716
} = primordials;
1817

1918
const {
@@ -22,15 +21,17 @@ const {
2221
ERR_INVALID_THIS,
2322
ERR_NO_ICU,
2423
} = require('internal/errors').codes;
24+
const kMethod = Symbol('method');
2525
const kHandle = Symbol('handle');
2626
const kFlags = Symbol('flags');
2727
const kEncoding = Symbol('encoding');
2828
const kDecoder = Symbol('decoder');
2929
const kFatal = Symbol('kFatal');
3030
const kUTF8FastPath = Symbol('kUTF8FastPath');
31-
const kWindows1252FastPath = Symbol('kWindows1252FastPath');
3231
const kIgnoreBOM = Symbol('kIgnoreBOM');
3332

33+
const { isSinglebyteEncoding, createSinglebyteDecoder } = require('internal/encoding/single-byte');
34+
3435
const {
3536
getConstructorOf,
3637
customInspectSymbol: inspect,
@@ -55,7 +56,6 @@ const {
5556
encodeIntoResults,
5657
encodeUtf8String,
5758
decodeUTF8,
58-
decodeWindows1252,
5959
} = binding;
6060

6161
const { Buffer } = require('buffer');
@@ -69,7 +69,7 @@ const CONVERTER_FLAGS_FLUSH = 0x1;
6969
const CONVERTER_FLAGS_FATAL = 0x2;
7070
const CONVERTER_FLAGS_IGNORE_BOM = 0x4;
7171

72-
const empty = new Uint8Array(0);
72+
const empty = Buffer.alloc(0);
7373

7474
const encodings = new SafeMap([
7575
['unicode-1-1-utf-8', 'utf-8'],
@@ -387,6 +387,24 @@ ObjectDefineProperties(
387387
[SymbolToStringTag]: { __proto__: null, configurable: true, value: 'TextEncoder' },
388388
});
389389

390+
function parseInput(input) {
391+
if (isAnyArrayBuffer(input)) {
392+
try {
393+
return Buffer.from(input);
394+
} catch {
395+
return empty;
396+
}
397+
} else if (isArrayBufferView(input)) {
398+
try {
399+
return Buffer.from(input.buffer, input.byteOffset, input.byteLength);
400+
} catch {
401+
return empty;
402+
}
403+
} else {
404+
throw new ERR_INVALID_ARG_TYPE('input', ['ArrayBuffer', 'ArrayBufferView'], input);
405+
}
406+
}
407+
390408
const TextDecoder =
391409
internalBinding('config').hasIntl ?
392410
makeTextDecoderICU() :
@@ -420,10 +438,12 @@ function makeTextDecoderICU() {
420438
this[kFatal] = Boolean(options?.fatal);
421439
// Only support fast path for UTF-8.
422440
this[kUTF8FastPath] = enc === 'utf-8';
423-
this[kWindows1252FastPath] = enc === 'windows-1252';
424441
this[kHandle] = undefined;
442+
this[kMethod] = undefined;
425443

426-
if (!this[kUTF8FastPath] && !this[kWindows1252FastPath]) {
444+
if (isSinglebyteEncoding(this.encoding)) {
445+
this[kMethod] = createSinglebyteDecoder(this.encoding, this[kFatal]);
446+
} else if (!this[kUTF8FastPath]) {
427447
this.#prepareConverter();
428448
}
429449
}
@@ -438,22 +458,18 @@ function makeTextDecoderICU() {
438458

439459
decode(input = empty, options = kEmptyObject) {
440460
validateDecoder(this);
461+
validateObject(options, 'options', kValidateObjectAllowObjectsAndNull);
462+
463+
if (this[kMethod]) return this[kMethod](parseInput(input));
441464

442465
this[kUTF8FastPath] &&= !(options?.stream);
443-
this[kWindows1252FastPath] &&= !(options?.stream);
444466

445467
if (this[kUTF8FastPath]) {
446468
return decodeUTF8(input, this[kIgnoreBOM], this[kFatal]);
447469
}
448470

449-
if (this[kWindows1252FastPath]) {
450-
return decodeWindows1252(input, this[kIgnoreBOM], this[kFatal]);
451-
}
452-
453471
this.#prepareConverter();
454472

455-
validateObject(options, 'options', kValidateObjectAllowObjectsAndNull);
456-
457473
let flags = 0;
458474
if (options !== null)
459475
flags |= options.stream ? 0 : CONVERTER_FLAGS_FLUSH;
@@ -476,7 +492,7 @@ function makeTextDecoderJS() {
476492
const kBOMSeen = Symbol('BOM seen');
477493

478494
function hasConverter(encoding) {
479-
return encoding === 'utf-8' || encoding === 'utf-16le';
495+
return encoding === 'utf-8' || encoding === 'utf-16le' || isSinglebyteEncoding(encoding);
480496
}
481497

482498
class TextDecoder {
@@ -502,30 +518,20 @@ function makeTextDecoderJS() {
502518
this[kFlags] = flags;
503519
this[kEncoding] = enc;
504520
this[kBOMSeen] = false;
521+
this[kMethod] = undefined;
522+
523+
if (isSinglebyteEncoding(this.encoding)) {
524+
this[kMethod] = createSinglebyteDecoder(this.encoding, this[kFatal]);
525+
}
505526
}
506527

507528
decode(input = empty, options = kEmptyObject) {
508529
validateDecoder(this);
509-
if (isAnyArrayBuffer(input)) {
510-
try {
511-
input = Buffer.from(input);
512-
} catch {
513-
input = empty;
514-
}
515-
} else if (isArrayBufferView(input)) {
516-
try {
517-
input = Buffer.from(input.buffer, input.byteOffset,
518-
input.byteLength);
519-
} catch {
520-
input = empty;
521-
}
522-
} else {
523-
throw new ERR_INVALID_ARG_TYPE('input',
524-
['ArrayBuffer', 'ArrayBufferView'],
525-
input);
526-
}
530+
input = parseInput(input);
527531
validateObject(options, 'options', kValidateObjectAllowObjectsAndNull);
528532

533+
if (this[kMethod]) return this[kMethod](input);
534+
529535
if (this[kFlags] & CONVERTER_FLAGS_FLUSH) {
530536
this[kBOMSeen] = false;
531537
}
Lines changed: 154 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,154 @@
1+
// Simplified version extracted from https://npmjs.com/package/@exodus/bytes codepath for 1-byte encodings
2+
// Copyright Exodus Movement. Licensed under MIT License.
3+
4+
'use strict';
5+
6+
const {
7+
ArrayFrom,
8+
ObjectKeys,
9+
ObjectPrototypeHasOwnProperty,
10+
SafeMap,
11+
SafeSet,
12+
StringPrototypeIncludes,
13+
TypedArrayFrom,
14+
Uint16Array,
15+
} = primordials;
16+
17+
const { Buffer, isAscii } = require('buffer');
18+
19+
const {
20+
ERR_ENCODING_NOT_SUPPORTED,
21+
ERR_ENCODING_INVALID_ENCODED_DATA,
22+
} = require('internal/errors').codes;
23+
24+
const { isBigEndian } = internalBinding('os');
25+
26+
/* fallback/single-byte.encodings.js, prepared differently */
27+
28+
const e = (length) => ArrayFrom({ length }, () => 1);
29+
const z = (length) => ArrayFrom({ length }, () => 0);
30+
31+
/* eslint-disable @stylistic/js/max-len */
32+
33+
// Common ranges
34+
35+
const k8a = [9472, 2, 10, 4, 4, 4, 4, 8, 8, 8, 8, 68, 4, 4, 4, 4, 1, 1, 1, -627, 640, -903, 1, 46, 28, 1, -8645, 8833, -8817, 2, 5, 64, 9305, 1, 1, -8449];
36+
const k8b = [-30, 1, 21, -18, 1, 15, -17, 18, -13, ...e(7), 16, -15, 1, 1, 1, -13, -4, 26, -1, -20, 17, 5, -4, -2, 3];
37+
const p1 = [8364, -8235, 8089, -7816, 7820, 8, -6, 1];
38+
const p2 = [-99, 12, 20, -12, 17, 37, -29, 2];
39+
const p3 = [1, 1, 65, -63, 158, -156, 1, 1, 1, 40, 30, 42, -46, 6, -66, 1, 83, -6, -6, -67, 176, ...p2, -114, 121, -119, 1, 1, 155, -49, 25, 16, -142, 159, 2, -158, 38, 42, -46, 6, -35, 1, 52, -6, -6, -36, 145, ...p2, -83, 90, -88, 1, 1, 124, -49, 25, 16, -111, 128, 2];
40+
const i0 = [128, ...e(32)];
41+
const i2 = [-40, -147, 1, 64, -62, 117, -51, -63, 69, -67, 79, -77, 79, -77, 1, 64, 2, 51, 4, -116, 1, 124, -122, 1, 129, 22, -148, 150, -148, 1, 133, -131, 118, -116, 1, 33, -31, 86, -51, -32, 38, -36, 48, -46, 48, -46, 1, 33, 2, 51, 4, -85, 1, 93, -91, 1, 98, 22, -117, 119, -117, 1, 102, 374];
42+
const i4a = [-75, -63, ...e(5), 104, -34, -67, 79, -77, 75, -73, 1];
43+
const i4b = [34, -32, ...e(5), 73, -34, -36, 48, -46, 44, -42, 1];
44+
const i7 = [721, 1, 1, -719, 721, -719, 721, ...e(19), 64604, -64602, ...e(43), 64559];
45+
const i8 = [...e(26), 64019, 0, -57327, 1, 57326];
46+
const w0 = [8364, -8235, 8089, -8087, 8091, 8, -6, 1, -8089, 8104];
47+
const w8 = [8072, 1, 3, 1, 5, -15, 1];
48+
const w1 = [...w8, -7480, 7750, -8129, 7897, -7911, -182];
49+
const w3 = [...w8, -8060, 8330, -8328, 8096, -8094];
50+
const m0 = [8558, -8328, 8374, -66, -8539, 16, 8043, -8070];
51+
52+
const encodings = {
53+
'__proto__': null,
54+
'ibm866': [1040, ...e(47), 8530, 1, 1, -145, 34, 61, 1, -12, -1, 14, -18, 6, 6, -1, -1, -75, 4, 32, -8, -16, -28, 60, 34, 1, -5, -6, 21, -3, -6, -16, 28, -5, 1, -4, 1, -12, -1, -6, 1, 24, -1, -82, -12, 124, -4, 8, 4, -16, -8512, ...e(15), -78, 80, -77, 80, -77, 80, -73, 80, -942, 8553, -8546, 8547, -260, -8306, 9468, -9472],
55+
'iso-8859-10': [...i0, 100, 14, 16, 8, -2, 14, -143, 148, -43, 80, 6, 23, -208, 189, -32, -154, 85, 14, 16, 8, -2, 14, -128, 133, -43, 80, 6, 23, 7831, -7850, -32, ...i4a, 1, 1, 117, 7, -121, 1, 1, 1, 146, -144, 154, -152, ...e(5), ...i4b, 1, 1, 86, 7, -90, 1, 1, 1, 115, -113, 123, -121, 1, 1, 1, 1, 58],
56+
'iso-8859-13': [...i0, 8061, -8059, 1, 1, 8058, -8056, 1, 49, -47, 173, -171, 1, 1, 1, 24, -22, 1, 1, 1, 8041, -8039, ...p3, 7835],
57+
'iso-8859-14': [...i0, 7522, 1, -7520, 103, 1, 7423, -7523, 7641, -7639, 7641, -119, 231, -7749, 1, 202, 7334, 1, -7423, 1, 7455, 1, -7563, 7584, 43, -42, 44, -35, 147, -111, 1, -36, -7585, ...e(15), 165, -163, ...e(5), 7572, -7570, ...e(5), 153, -151, ...e(16), 134, -132, ...e(5), 7541, -7539, ...e(5), 122],
58+
'iso-8859-15': [...i0, 1, 1, 1, 8201, -8199, 187, -185, 186, -184, ...e(10), 202, -200, 1, 1, 199, -197, 1, 1, 151, 1, 37],
59+
'iso-8859-16': [...i0, 100, 1, 60, 8043, -142, -7870, -185, 186, -184, 367, -365, 206, -204, 205, 1, -203, 1, 91, 54, 59, 7840, -8039, 1, 199, -113, 268, -350, 151, 1, 37, 4, -188, 1, 1, 64, -62, 66, -64, ...e(9), 65, 51, -113, 1, 1, 124, -122, 132, 22, -151, 1, 1, 1, 60, 258, -315, 1, 1, 1, 33, -31, 35, -33, ...e(9), 34, 51, -82, 1, 1, 93, -91, 101, 22, -120, 1, 1, 1, 29, 258],
60+
'iso-8859-2': [...i0, 100, 468, -407, -157, 153, 29, -179, 1, 184, -2, 6, 21, -204, 208, -2, -203, 85, 470, -409, -142, 138, 29, 364, -527, 169, -2, 6, 21, 355, -351, -2, ...i2],
61+
'iso-8859-3': [...i0, 134, 434, -565, 1, 65369, -65241, -125, 1, 136, 46, -64, 22, -135, 65360, -65154, -203, 119, -117, 1, 1, 1, 112, -110, 1, 121, 46, -64, 22, -120, 65344, -65153, -188, 1, 1, 65339, -65337, 70, -2, -65, ...e(8), 65326, -65324, 1, 1, 1, 76, -74, 1, 69, -67, 1, 1, 1, 144, -16, -125, 1, 1, 1, 65307, -65305, 39, -2, -34, ...e(8), 65294, -65292, 1, 1, 1, 45, -43, 1, 38, -36, 1, 1, 1, 113, -16, 380],
62+
'iso-8859-4': [...i0, 100, 52, 30, -178, 132, 19, -148, 1, 184, -78, 16, 68, -185, 208, -206, 1, 85, 470, -388, -163, 117, 19, 395, -527, 169, -78, 16, 68, -29, 52, -51, ...i4a, 92, -26, 53, 7, -22, -98, 1, 1, 1, 1, 154, -152, 1, 1, 140, 2, -139, ...i4b, 61, -26, 53, 7, -22, -67, 1, 1, 1, 1, 123, -121, 1, 1, 109, 2, 366],
63+
'iso-8859-5': [...i0, 865, ...e(11), -863, 865, ...e(65), 7367, -7365, ...e(11), -949, 951, 1],
64+
'iso-8859-6': [...i0, 65373, 0, 0, -65369, 65369, ...z(6), -63985, -1375, 65360, ...z(12), -63970, 63970, 0, 0, -63966, 63966, -63964, ...e(25), 63939, 0, 0, 0, 0, -63933, ...e(18), 63915, ...z(12)],
65+
'iso-8859-7': [...i0, 8056, 1, -8054, 8201, 3, -8201, 1, 1, 1, 721, -719, 1, 1, 65360, -57320, -8037, 1, 1, 1, 721, 1, 1, -719, ...i7],
66+
'iso-8859-8': [...i0, 65373, -65371, ...e(7), 46, -44, ...e(14), 62, -60, 1, 1, 1, 65343, ...z(31), -57318, -6727, ...i8],
67+
'koi8-r': [...k8a, 8450, ...e(14), -8544, 8545, ...e(10), -9411, 933, ...k8b, -28, ...k8b],
68+
'koi8-u': [...k8a, 3, 8448, -8446, 1, 8448, 1, 1, 1, 1, -8394, -51, 8448, 1, 1, 1, -8544, 3, 8543, -8541, 1, 8543, 1, 1, 1, 1, -8410, -130, -869, 933, ...k8b, -28, ...k8b],
69+
'macintosh': [196, 1, 2, 2, 8, 5, 6, 5, -1, 2, 2, -1, 2, 2, 2, -1, 2, 1, 2, -1, 2, 1, 2, 2, -1, 2, 2, -1, 5, -1, 2, 1, 7972, -8048, -14, 1, 4, 8059, -8044, 41, -49, -5, 8313, -8302, -12, 8632, -8602, 18, 8518, -8557, 8627, 1, -8640, 16, 8525, 15, -2, -7759, 7787, -8577, 16, 751, -707, 18, -57, -30, 11, ...m0, 32, 3, 18, 125, 1, 7872, 1, 8, 1, -5, 1, -7970, 9427, -9419, 121, 7884, 104, -115, 1, 56007, 1, -56033, -8042, 8035, 4, 18, -8046, 8, -9, 10, -3, 5, 1, 1, -3, 7, 1, 63531, -63533, 8, 1, -2, 88, 405, 22, -557, 553, 1, 1, -546, 549, -2, -20],
70+
'windows-1250': [...w0, -7888, 7897, -7903, 10, 25, -4, -233, ...w8, -8060, 8330, -8129, 7897, -7903, 10, 25, -4, -218, 551, 17, -407, -157, 96, -94, 1, 1, 1, 181, -179, 1, 1, 1, 205, -203, 1, 554, -409, -142, 1, 1, 1, 1, 77, 90, -164, 130, 416, -415, 62, ...i2],
71+
'windows-1251': [1026, 1, 7191, -7111, 7115, 8, -6, 1, 139, -124, -7207, 7216, -7215, 2, -1, 4, 67, 7110, 1, 3, 1, 5, -15, 1, -8060, 8330, -7369, 7137, -7136, 2, -1, 4, -959, 878, 80, -86, -868, 1004, -1002, 1, 858, -856, 859, -857, 1, 1, 1, 857, -855, 1, 853, 80, 59, -988, 1, 1, 922, 7365, -7362, -921, 925, -83, 80, 2, -71, ...e(63)],
72+
'windows-1252': [...p1, -7515, 7530, -7888, 7897, -7911, -197, 240, -238, 1, ...w1, 225, -6],
73+
'windows-1253': [...p1, -8089, 8104, -8102, 8111, -8109, 1, 1, 1, 1, ...w3, 1, 1, 1, 1, 741, 1, -739, 1, 1, 1, 1, 1, 1, 65364, -65362, 1, 1, 1, 8039, -8037, 1, 1, 1, 721, -719, 1, 1, ...i7],
74+
'windows-1254': [...p1, -7515, 7530, -7888, 7897, -7911, -197, 1, 1, 1, ...w1, 1, 218, -216, ...e(47), 79, -77, ...e(11), 84, 46, -127, ...e(16), 48, -46, ...e(11), 53, 46],
75+
'windows-1255': [...p1, -7515, 7530, -8102, 8111, -8109, 1, 1, 1, 1, ...w8, -7480, 7750, -8328, 8096, -8094, ...e(7), 8199, -8197, 1, 1, 1, 1, 46, -44, ...e(14), 62, -60, 1, 1, 1, 1, 1265, ...e(19), 45, 1, 1, 1, 1, 64009, ...z(6), -64045, ...i8],
76+
'windows-1256': [8364, -6702, 6556, -7816, 7820, 8, -6, 1, -7515, 7530, -6583, 6592, -7911, 1332, 18, -16, 39, 6505, 1, 3, 1, 5, -15, 1, -6507, 6777, -6801, 6569, -7911, 7865, 1, -6483, -1562, 1388, -1386, ...e(7), 1557, -1555, ...e(14), 1378, -1376, 1, 1, 1, 1377, 162, -160, ...e(21), -1375, 1376, 1, 1, 1, 6, 1, 1, 1, -1379, 1380, -1378, 1379, 1, 1, 1, -1377, 1, 1, 1, 1, 1374, 1, -1372, 1, 1372, 1, 1, 1, -1370, 1371, 1, -1369, 1370, -1368, 1369, -1367, 1, 7954, 1, -6461],
77+
'windows-1257': [...w0, -8102, 8111, -8109, 28, 543, -527, -40, ...w3, 19, 556, -572, 1, 65373, -65371, 1, 1, 65369, -65367, 1, 49, -47, 173, -171, 1, 1, 1, 24, -22, ...e(5), ...p3, 347],
78+
'windows-1258': [...p1, -7515, 7530, -8102, 8111, -7911, -197, 1, 1, 1, ...w8, -7480, 7750, -8328, 8096, -7911, -182, 1, 218, -216, ...e(34), 64, -62, ...e(7), 565, -563, 1, 1, 65, -63, 568, -566, 1, 204, -202, 1, 1, 1, 1, 1, 1, 211, 340, -548, 1, 1, 1, 33, -31, ...e(7), 534, -532, 1, 1, 34, -32, 562, -560, 1, 173, -171, 1, 1, 1, 1, 1, 1, 180, 7931],
79+
'windows-874': [8364, -8235, 1, 1, 1, 8098, -8096, ...e(10), ...w8, -8060, ...e(8), 3425, ...e(57), 61891, 0, 0, 0, -61886, ...e(28), 61858, 0, 0, 0],
80+
'x-mac-cyrillic': [1040, ...e(31), 7153, -8048, 992, -1005, 4, 8059, -8044, 848, -856, -5, 8313, -7456, 80, 7694, -7773, 80, 7627, -8557, 8627, 1, -7695, -929, 988, -137, -4, 80, -77, 80, -78, 80, -79, 80, -2, -83, -857, ...m0, 875, 80, -79, 80, -7, 7102, 1, 8, 1, -5, 1, -7970, 7975, -7184, 80, -79, 80, 7351, -7445, 80, -2, -31, ...e(30), 7262],
81+
};
82+
83+
/* eslint-enable @stylistic/js/max-len */
84+
85+
/* fallback/single-byte.js + /single-byte.node.js, simplified */
86+
87+
function getEncoding(encoding) {
88+
if (encoding === 'x-user-defined') {
89+
return TypedArrayFrom(Uint16Array, { length: 256 }, (_, i) => (i >= 0x80 ? 0xf700 + i : i));
90+
}
91+
92+
if (!ObjectPrototypeHasOwnProperty(encodings, encoding)) {
93+
throw new ERR_ENCODING_NOT_SUPPORTED(encoding);
94+
}
95+
96+
const map = TypedArrayFrom(Uint16Array, { length: 256 }, (_, i) => i); // Unicode subset
97+
let prev = 0;
98+
map.set(TypedArrayFrom(Uint16Array, encodings[encoding], (x) => (prev += x)), 128);
99+
return map;
100+
}
101+
102+
const supported = new SafeSet([...ObjectKeys(encodings), 'iso-8859-8-i', 'x-user-defined']);
103+
const isSinglebyteEncoding = (enc) => supported.has(enc);
104+
105+
const decodersLoose = new SafeMap();
106+
const decodersFatal = new SafeMap();
107+
108+
function createSinglebyteDecoder(encoding, fatal) {
109+
const id = encoding === 'iso-8859-8-i' ? 'iso-8859-8' : encoding;
110+
const decoders = fatal ? decodersFatal : decodersLoose
111+
const cached = decoders.get(id);
112+
if (cached) return cached;
113+
114+
const map = getEncoding(id);
115+
const incomplete = map.includes(0xfffd);
116+
117+
// Expecta type-checked Buffer input
118+
const decoder = (buf) => {
119+
if (buf.byteLength === 0) return '';
120+
if (isAscii(buf)) return buf.latin1Slice(); // .latin1Slice is faster than .asciiSlice
121+
const o = TypedArrayFrom(Uint16Array, buf); // Copy to modify in-place, also those are 16-bit now
122+
123+
let i = 0;
124+
for (const end7 = o.length - 7; i < end7; i += 8) {
125+
o[i] = map[o[i]];
126+
o[i + 1] = map[o[i + 1]];
127+
o[i + 2] = map[o[i + 2]];
128+
o[i + 3] = map[o[i + 3]];
129+
o[i + 4] = map[o[i + 4]];
130+
o[i + 5] = map[o[i + 5]];
131+
o[i + 6] = map[o[i + 6]];
132+
o[i + 7] = map[o[i + 7]];
133+
}
134+
135+
for (const end = o.length; i < end; i++) o[i] = map[o[i]];
136+
137+
const b = Buffer.from(o.buffer, o.byteOffset, o.byteLength);
138+
if (isBigEndian) b.swap16();
139+
const string = b.ucs2Slice();
140+
if (fatal && incomplete && StringPrototypeIncludes(string, '\uFFFD')) {
141+
throw new ERR_ENCODING_INVALID_ENCODED_DATA(encoding, undefined);
142+
}
143+
return string;
144+
};
145+
146+
decoders.set(id, decoder);
147+
return decoder;
148+
}
149+
150+
module.exports = {
151+
isSinglebyteEncoding,
152+
createSinglebyteDecoder,
153+
getEncoding, // for tests
154+
};

0 commit comments

Comments
 (0)