index.cjs 58 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998
  1. "use strict";
  2. var __create = Object.create;
  3. var __defProp = Object.defineProperty;
  4. var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
  5. var __getOwnPropNames = Object.getOwnPropertyNames;
  6. var __getProtoOf = Object.getPrototypeOf;
  7. var __hasOwnProp = Object.prototype.hasOwnProperty;
  8. var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
  9. var __export = (target, all) => {
  10. for (var name in all)
  11. __defProp(target, name, { get: all[name], enumerable: true });
  12. };
  13. var __copyProps = (to, from, except, desc) => {
  14. if (from && typeof from === "object" || typeof from === "function") {
  15. for (let key of __getOwnPropNames(from))
  16. if (!__hasOwnProp.call(to, key) && key !== except)
  17. __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
  18. }
  19. return to;
  20. };
  21. var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
  22. // If the importer is in node compatibility mode or this is not an ESM
  23. // file that has been converted to a CommonJS file using a Babel-
  24. // compatible transform (i.e. "__esModule" has not been set), then set
  25. // "default" to the CommonJS "module.exports" for node compatibility.
  26. isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
  27. mod
  28. ));
  29. var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
  30. // src/index.js
  31. var index_exports = {};
  32. __export(index_exports, {
  33. BodyParser: () => BodyParser,
  34. CHARACTER_ENCODING_LIST: () => CHARACTER_ENCODING_LIST,
  35. CookiesParser: () => CookiesParser,
  36. DataSender: () => DataSender,
  37. EXPOSED_ERROR_PROPERTIES: () => EXPOSED_ERROR_PROPERTIES,
  38. ErrorSender: () => ErrorSender,
  39. HookInvoker: () => HookInvoker,
  40. HookRegistry: () => HookRegistry,
  41. HttpMethod: () => HttpMethod,
  42. METHODS_WITH_BODY: () => METHODS_WITH_BODY,
  43. QueryParser: () => QueryParser,
  44. RequestContext: () => RequestContext,
  45. RequestParser: () => RequestParser,
  46. Route: () => Route,
  47. RouteRegistry: () => RouteRegistry,
  48. RouterHookType: () => RouterHookType,
  49. RouterOptions: () => RouterOptions,
  50. TrieRouter: () => TrieRouter,
  51. UNPARSABLE_MEDIA_TYPES: () => UNPARSABLE_MEDIA_TYPES,
  52. cloneDeep: () => cloneDeep,
  53. createCookieString: () => createCookieString,
  54. createError: () => createError,
  55. createRequestMock: () => createRequestMock,
  56. createResponseMock: () => createResponseMock,
  57. createRouteMock: () => createRouteMock,
  58. fetchRequestBody: () => fetchRequestBody,
  59. getRequestPathname: () => getRequestPathname,
  60. isPromise: () => isPromise,
  61. isReadableStream: () => isReadableStream,
  62. isResponseSent: () => isResponseSent,
  63. isWritableStream: () => isWritableStream,
  64. parseContentType: () => parseContentType,
  65. parseCookieString: () => parseCookieString,
  66. parseJsonBody: () => parseJsonBody,
  67. toCamelCase: () => toCamelCase
  68. });
  69. module.exports = __toCommonJS(index_exports);
  70. // src/route.js
  71. var import_js_debug = require("@e22m4u/js-debug");
  72. var import_js_format12 = require("@e22m4u/js-format");
  73. // src/hooks/hook-invoker.js
  74. var import_js_format11 = require("@e22m4u/js-format");
  75. // src/debuggable-service.js
  76. var import_js_service = require("@e22m4u/js-service");
  77. var MODULE_DEBUG_NAMESPACE = "jsTrieRouter";
  78. var _DebuggableService = class _DebuggableService extends import_js_service.DebuggableService {
  79. /**
  80. * Constructor.
  81. *
  82. * @param {ServiceContainer} container
  83. */
  84. constructor(container = void 0) {
  85. super(container, {
  86. namespace: MODULE_DEBUG_NAMESPACE,
  87. noEnvironmentNamespace: true
  88. });
  89. }
  90. };
  91. __name(_DebuggableService, "DebuggableService");
  92. var DebuggableService = _DebuggableService;
  93. // src/utils/clone-deep.js
  94. function cloneDeep(value) {
  95. if (value == null || typeof value !== "object") {
  96. return value;
  97. }
  98. if (value instanceof Date) {
  99. return new Date(value.getTime());
  100. }
  101. if (Array.isArray(value)) {
  102. return value.map((item) => cloneDeep(item));
  103. }
  104. const proto = Object.getPrototypeOf(value);
  105. if (proto === Object.prototype || proto === null) {
  106. const newObj = proto === null ? /* @__PURE__ */ Object.create(null) : {};
  107. for (const key in value) {
  108. if (Object.prototype.hasOwnProperty.call(value, key)) {
  109. newObj[key] = cloneDeep(value[key]);
  110. }
  111. }
  112. return newObj;
  113. }
  114. return value;
  115. }
  116. __name(cloneDeep, "cloneDeep");
  117. // src/utils/is-promise.js
  118. function isPromise(value) {
  119. if (!value) {
  120. return false;
  121. }
  122. if (typeof value !== "object") {
  123. return false;
  124. }
  125. return typeof value.then === "function";
  126. }
  127. __name(isPromise, "isPromise");
  128. // src/utils/create-error.js
  129. var import_js_format = require("@e22m4u/js-format");
  130. function createError(errorCtor, message, ...args) {
  131. if (typeof errorCtor !== "function") {
  132. throw new import_js_format.InvalidArgumentError(
  133. 'The first parameter of "createError" must be a constructor, but %v was given.',
  134. errorCtor
  135. );
  136. }
  137. if (message != null && typeof message !== "string") {
  138. throw new import_js_format.InvalidArgumentError(
  139. 'The second parameter of "createError" must be a String, but %v was given.',
  140. message
  141. );
  142. }
  143. if (message == null) {
  144. return new errorCtor();
  145. }
  146. const interpolatedMessage = (0, import_js_format.format)(message, ...args);
  147. return new errorCtor(interpolatedMessage);
  148. }
  149. __name(createError, "createError");
  150. // src/utils/to-camel-case.js
  151. var import_js_format2 = require("@e22m4u/js-format");
  152. function toCamelCase(input) {
  153. if (typeof input !== "string") {
  154. throw new import_js_format2.InvalidArgumentError(
  155. 'The first parameter of "toCamelCase" must be a String, but %v was given.',
  156. input
  157. );
  158. }
  159. return input.replace(/(^\w|[A-Z]|\b\w)/g, (c) => c.toUpperCase()).replace(/\W+/g, "").replace(/(^\w)/g, (c) => c.toLowerCase());
  160. }
  161. __name(toCamelCase, "toCamelCase");
  162. // src/utils/is-response-sent.js
  163. var import_js_format3 = require("@e22m4u/js-format");
  164. function isResponseSent(response) {
  165. if (!response || typeof response !== "object" || Array.isArray(response) || typeof response.headersSent !== "boolean") {
  166. throw new import_js_format3.InvalidArgumentError(
  167. 'The first parameter of "isResponseSent" must be an instance of ServerResponse, but %v was given.',
  168. response
  169. );
  170. }
  171. return response.headersSent;
  172. }
  173. __name(isResponseSent, "isResponseSent");
  174. // src/utils/create-route-mock.js
  175. function createRouteMock(options = {}) {
  176. return new Route({
  177. method: options.method || HttpMethod.GET,
  178. path: options.path || "/",
  179. handler: options.handler || (() => "OK")
  180. });
  181. }
  182. __name(createRouteMock, "createRouteMock");
  183. // src/utils/is-readable-stream.js
  184. function isReadableStream(value) {
  185. if (!value || typeof value !== "object") {
  186. return false;
  187. }
  188. return typeof value.pipe === "function";
  189. }
  190. __name(isReadableStream, "isReadableStream");
  191. // src/utils/parse-content-type.js
  192. var import_js_format4 = require("@e22m4u/js-format");
  193. function parseContentType(input) {
  194. if (typeof input !== "string") {
  195. throw new import_js_format4.InvalidArgumentError(
  196. "The first parameter of `parseContentType` must be a String, but %v was given.",
  197. input
  198. );
  199. }
  200. const res = { mediaType: void 0, charset: void 0, boundary: void 0 };
  201. const re = /^\s*([^\s;/]+\/[^\s;/]+)(?:;\s*charset=([^\s;]+))?(?:;\s*boundary=([^\s;]+))?.*$/i;
  202. const matches = re.exec(input);
  203. if (matches && matches[1]) {
  204. res.mediaType = matches[1];
  205. if (matches[2]) {
  206. res.charset = matches[2];
  207. }
  208. if (matches[3]) {
  209. res.boundary = matches[3];
  210. }
  211. }
  212. return res;
  213. }
  214. __name(parseContentType, "parseContentType");
  215. // src/utils/is-writable-stream.js
  216. function isWritableStream(value) {
  217. if (!value || typeof value !== "object") {
  218. return false;
  219. }
  220. return typeof value.end === "function";
  221. }
  222. __name(isWritableStream, "isWritableStream");
  223. // src/utils/fetch-request-body.js
  224. var import_http_errors = __toESM(require("http-errors"), 1);
  225. var import_http = require("http");
  226. var import_js_format5 = require("@e22m4u/js-format");
  227. var CHARACTER_ENCODING_LIST = [
  228. "ascii",
  229. "utf8",
  230. "utf-8",
  231. "utf16le",
  232. "utf-16le",
  233. "ucs2",
  234. "ucs-2",
  235. "latin1"
  236. ];
  237. function fetchRequestBody(request, bodyBytesLimit = 0) {
  238. if (!(request instanceof import_http.IncomingMessage)) {
  239. throw new import_js_format5.InvalidArgumentError(
  240. 'The first parameter of "fetchRequestBody" must be an IncomingMessage instance, but %v was given.',
  241. request
  242. );
  243. }
  244. if (typeof bodyBytesLimit !== "number") {
  245. throw new import_js_format5.InvalidArgumentError(
  246. 'The parameter "bodyBytesLimit" of "fetchRequestBody" must be a number, but %v was given.',
  247. bodyBytesLimit
  248. );
  249. }
  250. return new Promise((resolve, reject) => {
  251. const contentLength = parseInt(
  252. request.headers["content-length"] || "0",
  253. 10
  254. );
  255. if (bodyBytesLimit && contentLength && contentLength > bodyBytesLimit) {
  256. throw createError(
  257. import_http_errors.default.PayloadTooLarge,
  258. "Request body limit is %s bytes, but %s bytes given.",
  259. bodyBytesLimit,
  260. contentLength
  261. );
  262. }
  263. let encoding = "utf-8";
  264. const contentType = request.headers["content-type"] || "";
  265. if (contentType) {
  266. const parsedContentType = parseContentType(contentType);
  267. if (parsedContentType && parsedContentType.charset) {
  268. encoding = parsedContentType.charset.toLowerCase();
  269. if (!CHARACTER_ENCODING_LIST.includes(encoding)) {
  270. throw createError(
  271. import_http_errors.default.UnsupportedMediaType,
  272. "Request encoding %v is not supported.",
  273. encoding
  274. );
  275. }
  276. }
  277. }
  278. const data = [];
  279. let receivedLength = 0;
  280. const onData = /* @__PURE__ */ __name((chunk) => {
  281. receivedLength += chunk.length;
  282. if (bodyBytesLimit && receivedLength > bodyBytesLimit) {
  283. request.removeAllListeners();
  284. const error = createError(
  285. import_http_errors.default.PayloadTooLarge,
  286. "Request body limit is %v bytes, but %v bytes given.",
  287. bodyBytesLimit,
  288. receivedLength
  289. );
  290. reject(error);
  291. return;
  292. }
  293. data.push(chunk);
  294. }, "onData");
  295. const onEnd = /* @__PURE__ */ __name(() => {
  296. request.removeAllListeners();
  297. if (contentLength && contentLength !== receivedLength) {
  298. const error = createError(
  299. import_http_errors.default.BadRequest,
  300. 'Received bytes do not match the "content-length" header.'
  301. );
  302. reject(error);
  303. return;
  304. }
  305. const buffer = Buffer.concat(data);
  306. const body = buffer.toString(encoding);
  307. resolve(body || void 0);
  308. }, "onEnd");
  309. const onError = /* @__PURE__ */ __name((error) => {
  310. request.removeAllListeners();
  311. reject((0, import_http_errors.default)(400, error));
  312. }, "onError");
  313. request.on("data", onData);
  314. request.on("end", onEnd);
  315. request.on("error", onError);
  316. request.resume();
  317. });
  318. }
  319. __name(fetchRequestBody, "fetchRequestBody");
  320. // src/utils/parse-cookie-string.js
  321. var import_js_format6 = require("@e22m4u/js-format");
  322. function parseCookieString(input) {
  323. if (typeof input !== "string") {
  324. throw new import_js_format6.InvalidArgumentError(
  325. 'The first parameter of "parseCookieString" must be a String, but %v was given.',
  326. input
  327. );
  328. }
  329. return input.split(";").filter((v) => v !== "").map((v) => v.split("=")).reduce((cookies, tuple) => {
  330. const key = decodeURIComponent(tuple[0]).trim();
  331. const value = tuple[1] !== void 0 ? decodeURIComponent(tuple[1]).trim() : "";
  332. cookies[key] = value;
  333. return cookies;
  334. }, {});
  335. }
  336. __name(parseCookieString, "parseCookieString");
  337. // src/utils/create-request-mock.js
  338. var import_net = require("net");
  339. var import_tls = require("tls");
  340. var import_http2 = require("http");
  341. var import_querystring = __toESM(require("querystring"), 1);
  342. var import_js_format8 = require("@e22m4u/js-format");
  343. // src/utils/create-cookie-string.js
  344. var import_js_format7 = require("@e22m4u/js-format");
  345. function createCookieString(data) {
  346. if (!data || typeof data !== "object" || Array.isArray(data)) {
  347. throw new import_js_format7.InvalidArgumentError(
  348. 'The first parameter of "createCookieString" must be an Object, but %v was given.',
  349. data
  350. );
  351. }
  352. let cookies = "";
  353. for (const key in data) {
  354. if (!Object.prototype.hasOwnProperty.call(data, key)) {
  355. continue;
  356. }
  357. const val = data[key];
  358. if (val == null) {
  359. continue;
  360. }
  361. cookies += `${key}=${val}; `;
  362. }
  363. return cookies.trim();
  364. }
  365. __name(createCookieString, "createCookieString");
  366. // src/utils/create-request-mock.js
  367. function createRequestMock(patch) {
  368. if (patch != null && typeof patch !== "object" || Array.isArray(patch)) {
  369. throw new import_js_format8.InvalidArgumentError(
  370. 'The first parameter of "createRequestMock" must be an Object, but %v was given.',
  371. patch
  372. );
  373. }
  374. patch = patch || {};
  375. if (patch.host != null && typeof patch.host !== "string") {
  376. throw new import_js_format8.InvalidArgumentError(
  377. 'The parameter "host" of "createRequestMock" must be a String, but %v was given.',
  378. patch.host
  379. );
  380. }
  381. if (patch.method != null && typeof patch.method !== "string") {
  382. throw new import_js_format8.InvalidArgumentError(
  383. 'The parameter "method" of "createRequestMock" must be a String, but %v was given.',
  384. patch.method
  385. );
  386. }
  387. if (patch.secure != null && typeof patch.secure !== "boolean") {
  388. throw new import_js_format8.InvalidArgumentError(
  389. 'The parameter "secure" of "createRequestMock" must be a Boolean, but %v was given.',
  390. patch.secure
  391. );
  392. }
  393. if (patch.path != null && typeof patch.path !== "string") {
  394. throw new import_js_format8.InvalidArgumentError(
  395. 'The parameter "path" of "createRequestMock" must be a String, but %v was given.',
  396. patch.path
  397. );
  398. }
  399. if (patch.query != null && typeof patch.query !== "object" && typeof patch.query !== "string" || Array.isArray(patch.query)) {
  400. throw new import_js_format8.InvalidArgumentError(
  401. 'The parameter "query" of "createRequestMock" must be a String or Object, but %v was given.',
  402. patch.query
  403. );
  404. }
  405. if (patch.cookies != null && typeof patch.cookies !== "string" && typeof patch.cookies !== "object" || Array.isArray(patch.cookies)) {
  406. throw new import_js_format8.InvalidArgumentError(
  407. 'The parameter "cookies" of "createRequestMock" must be a String or Object, but %v was given.',
  408. patch.cookies
  409. );
  410. }
  411. if (patch.headers != null && typeof patch.headers !== "object" || Array.isArray(patch.headers)) {
  412. throw new import_js_format8.InvalidArgumentError(
  413. 'The parameter "headers" of "createRequestMock" must be an Object, but %v was given.',
  414. patch.headers
  415. );
  416. }
  417. if (patch.stream != null && !isReadableStream(patch.stream)) {
  418. throw new import_js_format8.InvalidArgumentError(
  419. 'The parameter "stream" of "createRequestMock" must be a Stream, but %v was given.',
  420. patch.stream
  421. );
  422. }
  423. if (patch.encoding != null) {
  424. if (typeof patch.encoding !== "string") {
  425. throw new import_js_format8.InvalidArgumentError(
  426. 'The parameter "encoding" of "createRequestMock" must be a String, but %v was given.',
  427. patch.encoding
  428. );
  429. }
  430. if (!CHARACTER_ENCODING_LIST.includes(patch.encoding)) {
  431. throw new import_js_format8.InvalidArgumentError(
  432. "Character encoding %v is not supported.",
  433. patch.encoding
  434. );
  435. }
  436. }
  437. if (patch.stream) {
  438. if (patch.secure != null) {
  439. throw new import_js_format8.InvalidArgumentError(
  440. 'The "createRequestMock" does not allow specifying the "stream" and "secure" options simultaneously.'
  441. );
  442. }
  443. if (patch.body != null) {
  444. throw new import_js_format8.InvalidArgumentError(
  445. 'The "createRequestMock" does not allow specifying the "stream" and "body" options simultaneously.'
  446. );
  447. }
  448. if (patch.encoding != null) {
  449. throw new import_js_format8.InvalidArgumentError(
  450. 'The "createRequestMock" does not allow specifying the "stream" and "encoding" options simultaneously.'
  451. );
  452. }
  453. }
  454. const request = patch.stream || createRequestStream(patch.secure, patch.body, patch.encoding);
  455. request.url = createRequestUrl(patch.path || "/", patch.query);
  456. request.headers = createRequestHeaders(
  457. patch.host,
  458. patch.secure,
  459. patch.body,
  460. patch.cookies,
  461. patch.encoding,
  462. patch.headers
  463. );
  464. request.method = (patch.method || "get").toUpperCase();
  465. return request;
  466. }
  467. __name(createRequestMock, "createRequestMock");
  468. function createRequestStream(secure, body, encoding) {
  469. if (encoding != null && typeof encoding !== "string") {
  470. throw new import_js_format8.InvalidArgumentError(
  471. 'The parameter "encoding" of "createRequestStream" must be a String, but %v was given.',
  472. encoding
  473. );
  474. }
  475. encoding = encoding || "utf-8";
  476. let socket = new import_net.Socket();
  477. if (secure) {
  478. socket = new import_tls.TLSSocket(socket);
  479. }
  480. const request = new import_http2.IncomingMessage(socket);
  481. if (body != null) {
  482. if (typeof body === "string") {
  483. request.push(body, encoding);
  484. } else if (Buffer.isBuffer(body)) {
  485. request.push(body);
  486. } else {
  487. request.push(JSON.stringify(body));
  488. }
  489. }
  490. request.push(null);
  491. return request;
  492. }
  493. __name(createRequestStream, "createRequestStream");
  494. function createRequestUrl(path, query) {
  495. if (typeof path !== "string") {
  496. throw new import_js_format8.InvalidArgumentError(
  497. 'The parameter "path" of "createRequestUrl" must be a String, but %v was given.',
  498. path
  499. );
  500. }
  501. if (query != null && typeof query !== "string" && typeof query !== "object" || Array.isArray(query)) {
  502. throw new import_js_format8.InvalidArgumentError(
  503. 'The parameter "query" of "createRequestUrl" must be a String or Object, but %v was given.',
  504. query
  505. );
  506. }
  507. let url = ("/" + path).replace("//", "/");
  508. if (typeof query === "object") {
  509. const qs = import_querystring.default.stringify(query);
  510. if (qs) {
  511. url += `?${qs}`;
  512. }
  513. } else if (typeof query === "string") {
  514. url += `?${query.replace(/^\?/, "")}`;
  515. }
  516. return url;
  517. }
  518. __name(createRequestUrl, "createRequestUrl");
  519. function createRequestHeaders(host, secure, body, cookies, encoding, headers) {
  520. if (host != null && typeof host !== "string") {
  521. throw new import_js_format8.InvalidArgumentError(
  522. 'The parameter "host" of "createRequestHeaders" a non-empty String, but %v was given.',
  523. host
  524. );
  525. }
  526. host = host || "localhost";
  527. if (secure != null && typeof secure !== "boolean") {
  528. throw new import_js_format8.InvalidArgumentError(
  529. 'The parameter "secure" of "createRequestHeaders" must be a String, but %v was given.',
  530. secure
  531. );
  532. }
  533. secure = Boolean(secure);
  534. if (cookies != null && typeof cookies !== "object" && typeof cookies !== "string" || Array.isArray(cookies)) {
  535. throw new import_js_format8.InvalidArgumentError(
  536. 'The parameter "cookies" of "createRequestHeaders" must be a String or Object, but %v was given.',
  537. cookies
  538. );
  539. }
  540. if (headers != null && typeof headers !== "object" || Array.isArray(headers)) {
  541. throw new import_js_format8.InvalidArgumentError(
  542. 'The parameter "headers" of "createRequestHeaders" must be an Object, but %v was given.',
  543. headers
  544. );
  545. }
  546. headers = headers || {};
  547. if (encoding != null && typeof encoding !== "string") {
  548. throw new import_js_format8.InvalidArgumentError(
  549. 'The parameter "encoding" of "createRequestHeaders" must be a String, but %v was given.',
  550. encoding
  551. );
  552. }
  553. encoding = encoding || "utf-8";
  554. const obj = { ...headers };
  555. obj["host"] = host;
  556. if (secure) {
  557. obj["x-forwarded-proto"] = "https";
  558. }
  559. if (cookies != null) {
  560. if (typeof cookies === "string") {
  561. obj["cookie"] = obj["cookie"] ? obj["cookie"] : "";
  562. obj["cookie"] += obj["cookie"] ? `; ${cookies}` : cookies;
  563. } else if (typeof cookies === "object") {
  564. obj["cookie"] = obj["cookie"] ? obj["cookie"] : "";
  565. const newCookies = createCookieString(cookies);
  566. obj["cookie"] += obj["cookie"] ? `; ${newCookies}` : newCookies;
  567. }
  568. }
  569. if (obj["content-type"] == null) {
  570. if (typeof body === "string") {
  571. obj["content-type"] = "text/plain";
  572. } else if (Buffer.isBuffer(body)) {
  573. obj["content-type"] = "application/octet-stream";
  574. } else if (typeof body === "object" || typeof body === "boolean" || typeof body === "number") {
  575. obj["content-type"] = "application/json";
  576. }
  577. }
  578. if (body != null && obj["content-length"] == null) {
  579. if (typeof body === "string") {
  580. const length = Buffer.byteLength(body, encoding);
  581. obj["content-length"] = String(length);
  582. } else if (Buffer.isBuffer(body)) {
  583. const length = Buffer.byteLength(body);
  584. obj["content-length"] = String(length);
  585. } else if (typeof body === "object" || typeof body === "boolean" || typeof body === "number") {
  586. const json = JSON.stringify(body);
  587. const length = Buffer.byteLength(json, encoding);
  588. obj["content-length"] = String(length);
  589. }
  590. }
  591. return obj;
  592. }
  593. __name(createRequestHeaders, "createRequestHeaders");
  594. // src/utils/create-response-mock.js
  595. var import_stream = require("stream");
  596. function createResponseMock() {
  597. const response = new import_stream.PassThrough();
  598. patchEncoding(response);
  599. patchHeaders(response);
  600. patchBody(response);
  601. return response;
  602. }
  603. __name(createResponseMock, "createResponseMock");
  604. function patchEncoding(response) {
  605. Object.defineProperty(response, "_encoding", {
  606. configurable: true,
  607. writable: true,
  608. value: void 0
  609. });
  610. Object.defineProperty(response, "setEncoding", {
  611. configurable: true,
  612. value: /* @__PURE__ */ __name(function(enc) {
  613. this._encoding = enc;
  614. return this;
  615. }, "value")
  616. });
  617. Object.defineProperty(response, "getEncoding", {
  618. configurable: true,
  619. value: /* @__PURE__ */ __name(function() {
  620. return this._encoding;
  621. }, "value")
  622. });
  623. }
  624. __name(patchEncoding, "patchEncoding");
  625. function patchHeaders(response) {
  626. Object.defineProperty(response, "_headersSent", {
  627. configurable: true,
  628. writable: true,
  629. value: false
  630. });
  631. Object.defineProperty(response, "headersSent", {
  632. configurable: true,
  633. get() {
  634. return this._headersSent;
  635. }
  636. });
  637. Object.defineProperty(response, "_headers", {
  638. configurable: true,
  639. writable: true,
  640. value: {}
  641. });
  642. Object.defineProperty(response, "setHeader", {
  643. configurable: true,
  644. value: /* @__PURE__ */ __name(function(name, value) {
  645. if (this.headersSent) {
  646. throw new Error(
  647. "Error [ERR_HTTP_HEADERS_SENT]: Cannot set headers after they are sent to the client"
  648. );
  649. }
  650. const key = name.toLowerCase();
  651. this._headers[key] = String(value);
  652. return this;
  653. }, "value")
  654. });
  655. Object.defineProperty(response, "getHeader", {
  656. configurable: true,
  657. value: /* @__PURE__ */ __name(function(name) {
  658. return this._headers[name.toLowerCase()];
  659. }, "value")
  660. });
  661. Object.defineProperty(response, "getHeaders", {
  662. configurable: true,
  663. value: /* @__PURE__ */ __name(function() {
  664. return JSON.parse(JSON.stringify(this._headers));
  665. }, "value")
  666. });
  667. }
  668. __name(patchHeaders, "patchHeaders");
  669. function patchBody(response) {
  670. let resolve, reject;
  671. const promise = new Promise((rsv, rej) => {
  672. resolve = rsv;
  673. reject = rej;
  674. });
  675. const data = [];
  676. response.on("data", (c) => data.push(c));
  677. response.on("error", (e) => reject(e));
  678. response.on("end", () => {
  679. resolve(Buffer.concat(data));
  680. });
  681. const originalEnd = response.end.bind(response);
  682. response.end = function(...args) {
  683. this._headersSent = true;
  684. return originalEnd(...args);
  685. };
  686. Object.defineProperty(response, "getBody", {
  687. configurable: true,
  688. value: /* @__PURE__ */ __name(function() {
  689. return promise.then((buffer) => {
  690. const enc = this.getEncoding();
  691. const str = buffer.toString(enc);
  692. return data.length ? str : void 0;
  693. });
  694. }, "value")
  695. });
  696. }
  697. __name(patchBody, "patchBody");
  698. // src/utils/get-request-pathname.js
  699. var import_js_format9 = require("@e22m4u/js-format");
  700. function getRequestPathname(request) {
  701. if (!request || typeof request !== "object" || Array.isArray(request) || typeof request.url !== "string") {
  702. throw new import_js_format9.InvalidArgumentError(
  703. 'The first parameter of "getRequestPathname" must be an instance of IncomingMessage, but %v was given.',
  704. request
  705. );
  706. }
  707. return (request.url || "/").replace(/\?.*$/, "");
  708. }
  709. __name(getRequestPathname, "getRequestPathname");
  710. // src/hooks/hook-registry.js
  711. var import_js_format10 = require("@e22m4u/js-format");
  712. var RouterHookType = {
  713. PRE_HANDLER: "preHandler",
  714. POST_HANDLER: "postHandler"
  715. };
  716. var _HookRegistry = class _HookRegistry {
  717. /**
  718. * Hooks.
  719. *
  720. * @type {Map<string, Function[]>}
  721. * @private
  722. */
  723. _hooks = /* @__PURE__ */ new Map();
  724. /**
  725. * Add hook.
  726. *
  727. * @param {string} type
  728. * @param {Function} hook
  729. * @returns {this}
  730. */
  731. addHook(type, hook) {
  732. if (!type || typeof type !== "string") {
  733. throw new import_js_format10.InvalidArgumentError(
  734. "The hook type is required, but %v was given.",
  735. type
  736. );
  737. }
  738. if (!Object.values(RouterHookType).includes(type)) {
  739. throw new import_js_format10.InvalidArgumentError(
  740. "The hook type %v is not supported.",
  741. type
  742. );
  743. }
  744. if (!hook || typeof hook !== "function") {
  745. throw new import_js_format10.InvalidArgumentError(
  746. "The hook %v must be a Function, but %v was given.",
  747. type,
  748. hook
  749. );
  750. }
  751. const hooks = this._hooks.get(type) || [];
  752. hooks.push(hook);
  753. this._hooks.set(type, hooks);
  754. return this;
  755. }
  756. /**
  757. * Has hook.
  758. *
  759. * @param {string} type
  760. * @param {Function} hook
  761. * @returns {boolean}
  762. */
  763. hasHook(type, hook) {
  764. if (!type || typeof type !== "string") {
  765. throw new import_js_format10.InvalidArgumentError(
  766. "The hook type is required, but %v was given.",
  767. type
  768. );
  769. }
  770. if (!Object.values(RouterHookType).includes(type)) {
  771. throw new import_js_format10.InvalidArgumentError(
  772. "The hook type %v is not supported.",
  773. type
  774. );
  775. }
  776. if (!hook || typeof hook !== "function") {
  777. throw new import_js_format10.InvalidArgumentError(
  778. "The hook %v must be a Function, but %v was given.",
  779. type,
  780. hook
  781. );
  782. }
  783. const hooks = this._hooks.get(type) || [];
  784. return hooks.indexOf(hook) > -1;
  785. }
  786. /**
  787. * Get hooks.
  788. *
  789. * @param {string} type
  790. * @returns {Function[]}
  791. */
  792. getHooks(type) {
  793. if (!type || typeof type !== "string") {
  794. throw new import_js_format10.InvalidArgumentError(
  795. "The hook type is required, but %v was given.",
  796. type
  797. );
  798. }
  799. if (!Object.values(RouterHookType).includes(type)) {
  800. throw new import_js_format10.InvalidArgumentError(
  801. "The hook type %v is not supported.",
  802. type
  803. );
  804. }
  805. return this._hooks.get(type) || [];
  806. }
  807. };
  808. __name(_HookRegistry, "HookRegistry");
  809. var HookRegistry = _HookRegistry;
  810. // src/hooks/hook-invoker.js
  811. var _HookInvoker = class _HookInvoker extends DebuggableService {
  812. /**
  813. * Invoke and continue until value received.
  814. *
  815. * @param {Route} route
  816. * @param {string} hookType
  817. * @param {import('http').ServerResponse} response
  818. * @param {*[]} args
  819. * @returns {Promise<*>|*}
  820. */
  821. invokeAndContinueUntilValueReceived(route, hookType, response, ...args) {
  822. if (!route || !(route instanceof Route)) {
  823. throw new import_js_format11.InvalidArgumentError(
  824. 'The parameter "route" of the HookInvoker.invokeAndContinueUntilValueReceived must be a Route instance, but %v was given.',
  825. route
  826. );
  827. }
  828. if (!hookType || typeof hookType !== "string") {
  829. throw new import_js_format11.InvalidArgumentError(
  830. 'The parameter "hookType" of the HookInvoker.invokeAndContinueUntilValueReceived must be a non-empty String, but %v was given.',
  831. hookType
  832. );
  833. }
  834. if (!Object.values(RouterHookType).includes(hookType)) {
  835. throw new import_js_format11.InvalidArgumentError(
  836. "The hook type %v is not supported.",
  837. hookType
  838. );
  839. }
  840. if (!response || typeof response !== "object" || Array.isArray(response) || typeof response.headersSent !== "boolean") {
  841. throw new import_js_format11.InvalidArgumentError(
  842. 'The parameter "response" of the HookInvoker.invokeAndContinueUntilValueReceived must be a ServerResponse instance, but %v was given.',
  843. response
  844. );
  845. }
  846. if (isResponseSent(response)) {
  847. return response;
  848. }
  849. const hooks = [
  850. ...this.getService(HookRegistry).getHooks(hookType),
  851. ...route.hookRegistry.getHooks(hookType)
  852. ];
  853. let result = void 0;
  854. for (let i = 0; i < hooks.length; i++) {
  855. const hook = hooks[i];
  856. result = hook(...args);
  857. if (isResponseSent(response)) {
  858. return response;
  859. }
  860. if (result != null) {
  861. if (isPromise(result)) {
  862. return (async () => {
  863. let asyncResult = await result;
  864. if (isResponseSent(response)) {
  865. return response;
  866. }
  867. if (asyncResult != null) {
  868. return asyncResult;
  869. }
  870. for (let j = i + 1; j < hooks.length; j++) {
  871. asyncResult = await hooks[j](...args);
  872. if (isResponseSent(response)) {
  873. return response;
  874. }
  875. if (asyncResult != null) {
  876. return asyncResult;
  877. }
  878. }
  879. return;
  880. })();
  881. }
  882. return result;
  883. }
  884. }
  885. return;
  886. }
  887. };
  888. __name(_HookInvoker, "HookInvoker");
  889. var HookInvoker = _HookInvoker;
  890. // src/route.js
  891. var HttpMethod = {
  892. GET: "GET",
  893. POST: "POST",
  894. PUT: "PUT",
  895. PATCH: "PATCH",
  896. DELETE: "DELETE"
  897. };
  898. var _Route = class _Route extends import_js_debug.Debuggable {
  899. /**
  900. * Method.
  901. *
  902. * @type {string}
  903. * @private
  904. */
  905. _method;
  906. /**
  907. * Getter of the method.
  908. *
  909. * @returns {string}
  910. */
  911. get method() {
  912. return this._method;
  913. }
  914. /**
  915. * Path template.
  916. *
  917. * @type {string}
  918. * @private
  919. */
  920. _path;
  921. /**
  922. * Getter of the path.
  923. *
  924. * @returns {string}
  925. */
  926. get path() {
  927. return this._path;
  928. }
  929. /**
  930. * Meta.
  931. *
  932. * @type {object}
  933. */
  934. _meta = {};
  935. /**
  936. * Getter of the meta.
  937. *
  938. * @returns {object}
  939. */
  940. get meta() {
  941. return this._meta;
  942. }
  943. /**
  944. * Handler.
  945. *
  946. * @type {RouteHandler}
  947. * @private
  948. */
  949. _handler;
  950. /**
  951. * Getter of the handler.
  952. *
  953. * @returns {*}
  954. */
  955. get handler() {
  956. return this._handler;
  957. }
  958. /**
  959. * Hook registry.
  960. *
  961. * @type {HookRegistry}
  962. * @private
  963. */
  964. _hookRegistry = new HookRegistry();
  965. /**
  966. * Getter of the hook registry.
  967. *
  968. * @returns {HookRegistry}
  969. */
  970. get hookRegistry() {
  971. return this._hookRegistry;
  972. }
  973. /**
  974. * Constructor.
  975. *
  976. * @param {RouteDefinition} routeDef
  977. */
  978. constructor(routeDef) {
  979. super({
  980. namespace: MODULE_DEBUG_NAMESPACE,
  981. noEnvironmentNamespace: true,
  982. noInstantiationMessage: true
  983. });
  984. if (!routeDef || typeof routeDef !== "object" || Array.isArray(routeDef)) {
  985. throw new import_js_format12.InvalidArgumentError(
  986. "The first parameter of Route.constructor must be an Object, but %v was given.",
  987. routeDef
  988. );
  989. }
  990. if (!routeDef.method || typeof routeDef.method !== "string") {
  991. throw new import_js_format12.InvalidArgumentError(
  992. 'The option "method" of the Route must be a non-empty String, but %v was given.',
  993. routeDef.method
  994. );
  995. }
  996. this._method = routeDef.method.toUpperCase();
  997. if (typeof routeDef.path !== "string") {
  998. throw new import_js_format12.InvalidArgumentError(
  999. 'The option "path" of the Route must be a String, but %v was given.',
  1000. routeDef.path
  1001. );
  1002. }
  1003. this._path = routeDef.path;
  1004. if (typeof routeDef.handler !== "function") {
  1005. throw new import_js_format12.InvalidArgumentError(
  1006. 'The option "handler" of the Route must be a Function, but %v was given.',
  1007. routeDef.handler
  1008. );
  1009. }
  1010. if (routeDef.meta != null) {
  1011. if (typeof routeDef.meta !== "object" || Array.isArray(routeDef.meta)) {
  1012. throw new import_js_format12.InvalidArgumentError(
  1013. 'The option "meta" of the Route must be a plain Object, but %v was given.',
  1014. routeDef.meta
  1015. );
  1016. }
  1017. this._meta = cloneDeep(routeDef.meta);
  1018. }
  1019. this._handler = routeDef.handler;
  1020. if (routeDef.preHandler != null) {
  1021. const preHandlerHooks = Array.isArray(routeDef.preHandler) ? routeDef.preHandler : [routeDef.preHandler];
  1022. preHandlerHooks.forEach((hook) => {
  1023. this._hookRegistry.addHook(RouterHookType.PRE_HANDLER, hook);
  1024. });
  1025. }
  1026. if (routeDef.postHandler != null) {
  1027. const postHandlerHooks = Array.isArray(routeDef.postHandler) ? routeDef.postHandler : [routeDef.postHandler];
  1028. postHandlerHooks.forEach((hook) => {
  1029. this._hookRegistry.addHook(RouterHookType.POST_HANDLER, hook);
  1030. });
  1031. }
  1032. this.ctorDebug("A new route %s %v was created.", this._method, this._path);
  1033. }
  1034. /**
  1035. * Handle request.
  1036. *
  1037. * @param {RequestContext} context
  1038. * @returns {*}
  1039. */
  1040. handle(context) {
  1041. const debug = this.getDebuggerFor(this.handle);
  1042. const requestPath = getRequestPathname(context.request);
  1043. debug(
  1044. "Invoking the Route handler for the request %s %v.",
  1045. this.method.toUpperCase(),
  1046. requestPath
  1047. );
  1048. return this._handler(context);
  1049. }
  1050. };
  1051. __name(_Route, "Route");
  1052. var Route = _Route;
  1053. // src/parsers/body-parser.js
  1054. var import_http_errors2 = __toESM(require("http-errors"), 1);
  1055. // src/router-options.js
  1056. var import_js_format13 = require("@e22m4u/js-format");
  1057. var _RouterOptions = class _RouterOptions extends DebuggableService {
  1058. /**
  1059. * Request body bytes limit.
  1060. *
  1061. * @type {number}
  1062. * @private
  1063. */
  1064. _requestBodyBytesLimit = 512e3;
  1065. // 512kb
  1066. /**
  1067. * Getter of request body bytes limit.
  1068. *
  1069. * @returns {number}
  1070. */
  1071. get requestBodyBytesLimit() {
  1072. return this._requestBodyBytesLimit;
  1073. }
  1074. /**
  1075. * Set request body bytes limit.
  1076. *
  1077. * @param {number} input
  1078. * @returns {RouterOptions}
  1079. */
  1080. setRequestBodyBytesLimit(input) {
  1081. if (typeof input !== "number" || input < 0) {
  1082. throw new import_js_format13.InvalidArgumentError(
  1083. 'The option "requestBodyBytesLimit" must be a positive Number or 0, but %v was given.',
  1084. input
  1085. );
  1086. }
  1087. this._requestBodyBytesLimit = input;
  1088. return this;
  1089. }
  1090. };
  1091. __name(_RouterOptions, "RouterOptions");
  1092. var RouterOptions = _RouterOptions;
  1093. // src/parsers/body-parser.js
  1094. var import_js_format14 = require("@e22m4u/js-format");
  1095. var METHODS_WITH_BODY = ["POST", "PUT", "PATCH", "DELETE"];
  1096. var UNPARSABLE_MEDIA_TYPES = ["multipart/form-data"];
  1097. var _BodyParser = class _BodyParser extends DebuggableService {
  1098. /**
  1099. * Parsers.
  1100. *
  1101. * @type {{[mime: string]: Function}}
  1102. */
  1103. _parsers = {
  1104. "text/plain": /* @__PURE__ */ __name((v) => String(v), "text/plain"),
  1105. "application/json": parseJsonBody
  1106. };
  1107. /**
  1108. * Set parser.
  1109. *
  1110. * @param {string} mediaType
  1111. * @param {Function} parser
  1112. * @returns {this}
  1113. */
  1114. defineParser(mediaType, parser) {
  1115. if (!mediaType || typeof mediaType !== "string") {
  1116. throw new import_js_format14.InvalidArgumentError(
  1117. 'The parameter "mediaType" of BodyParser.defineParser must be a non-empty String, but %v was given.',
  1118. mediaType
  1119. );
  1120. }
  1121. if (!parser || typeof parser !== "function") {
  1122. throw new import_js_format14.InvalidArgumentError(
  1123. 'The parameter "parser" of BodyParser.defineParser must be a Function, but %v was given.',
  1124. parser
  1125. );
  1126. }
  1127. this._parsers[mediaType] = parser;
  1128. return this;
  1129. }
  1130. /**
  1131. * Has parser.
  1132. *
  1133. * @param {string} mediaType
  1134. * @returns {boolean}
  1135. */
  1136. hasParser(mediaType) {
  1137. if (!mediaType || typeof mediaType !== "string") {
  1138. throw new import_js_format14.InvalidArgumentError(
  1139. 'The parameter "mediaType" of BodyParser.hasParser must be a non-empty String, but %v was given.',
  1140. mediaType
  1141. );
  1142. }
  1143. return Boolean(this._parsers[mediaType]);
  1144. }
  1145. /**
  1146. * Delete parser.
  1147. *
  1148. * @param {string} mediaType
  1149. * @returns {this}
  1150. */
  1151. deleteParser(mediaType) {
  1152. if (!mediaType || typeof mediaType !== "string") {
  1153. throw new import_js_format14.InvalidArgumentError(
  1154. 'The parameter "mediaType" of BodyParser.deleteParser must be a non-empty String, but %v was given.',
  1155. mediaType
  1156. );
  1157. }
  1158. const parser = this._parsers[mediaType];
  1159. if (!parser) {
  1160. throw new import_js_format14.InvalidArgumentError(
  1161. "The parser of %v is not found.",
  1162. mediaType
  1163. );
  1164. }
  1165. delete this._parsers[mediaType];
  1166. return this;
  1167. }
  1168. /**
  1169. * Parse.
  1170. *
  1171. * @param {import('http').IncomingMessage} request
  1172. * @returns {Promise<*>|undefined}
  1173. */
  1174. parse(request) {
  1175. const debug = this.getDebuggerFor(this.parse);
  1176. if (!METHODS_WITH_BODY.includes(request.method.toUpperCase())) {
  1177. debug(
  1178. "Body parsing was skipped for the %s request.",
  1179. request.method.toUpperCase()
  1180. );
  1181. return;
  1182. }
  1183. const contentType = (request.headers["content-type"] || "").replace(
  1184. /^([^;]+);.*$/,
  1185. "$1"
  1186. );
  1187. if (!contentType) {
  1188. debug(
  1189. "Body parsing was skipped because the request had no content type."
  1190. );
  1191. return;
  1192. }
  1193. const { mediaType } = parseContentType(contentType);
  1194. if (!mediaType) {
  1195. throw createError(
  1196. import_http_errors2.default.BadRequest,
  1197. 'Unable to parse the "content-type" header.'
  1198. );
  1199. }
  1200. const parser = this._parsers[mediaType];
  1201. if (!parser) {
  1202. if (UNPARSABLE_MEDIA_TYPES.includes(mediaType)) {
  1203. debug("Body parsing was skipped for %v.", mediaType);
  1204. return;
  1205. }
  1206. throw createError(
  1207. import_http_errors2.default.UnsupportedMediaType,
  1208. "Media type %v is not supported.",
  1209. mediaType
  1210. );
  1211. }
  1212. const bodyBytesLimit = this.getService(RouterOptions).requestBodyBytesLimit;
  1213. return fetchRequestBody(request, bodyBytesLimit).then((rawBody) => {
  1214. if (rawBody != null) {
  1215. return parser(rawBody);
  1216. }
  1217. return rawBody;
  1218. });
  1219. }
  1220. };
  1221. __name(_BodyParser, "BodyParser");
  1222. var BodyParser = _BodyParser;
  1223. function parseJsonBody(input) {
  1224. if (typeof input !== "string") {
  1225. return void 0;
  1226. }
  1227. try {
  1228. return JSON.parse(input);
  1229. } catch (error) {
  1230. throw createError(import_http_errors2.default.BadRequest, error.message);
  1231. }
  1232. }
  1233. __name(parseJsonBody, "parseJsonBody");
  1234. // src/parsers/query-parser.js
  1235. var import_querystring2 = __toESM(require("querystring"), 1);
  1236. var _QueryParser = class _QueryParser extends DebuggableService {
  1237. /**
  1238. * Parse
  1239. *
  1240. * @param {import('http').IncomingMessage} request
  1241. * @returns {object}
  1242. */
  1243. parse(request) {
  1244. const debug = this.getDebuggerFor(this.parse);
  1245. const queryStr = request.url.replace(/^[^?]*\??/, "");
  1246. const query = queryStr ? import_querystring2.default.parse(queryStr) : {};
  1247. const queryKeys = Object.keys(query);
  1248. if (queryKeys.length) {
  1249. queryKeys.forEach((key) => {
  1250. debug("The query parameter %v had the value %v.", key, query[key]);
  1251. });
  1252. } else {
  1253. debug(
  1254. "The request %s %v had no query parameters.",
  1255. request.method,
  1256. getRequestPathname(request)
  1257. );
  1258. }
  1259. return query;
  1260. }
  1261. };
  1262. __name(_QueryParser, "QueryParser");
  1263. var QueryParser = _QueryParser;
  1264. // src/parsers/cookies-parser.js
  1265. var _CookiesParser = class _CookiesParser extends DebuggableService {
  1266. /**
  1267. * Parse
  1268. *
  1269. * @param {import('http').IncomingMessage} request
  1270. * @returns {object}
  1271. */
  1272. parse(request) {
  1273. const debug = this.getDebuggerFor(this.parse);
  1274. const cookiesString = request.headers["cookie"] || "";
  1275. const cookies = parseCookieString(cookiesString);
  1276. const cookiesKeys = Object.keys(cookies);
  1277. if (cookiesKeys.length) {
  1278. cookiesKeys.forEach((key) => {
  1279. debug("The cookie %v had the value %v.", key, cookies[key]);
  1280. });
  1281. } else {
  1282. debug(
  1283. "The request %s %v had no cookies.",
  1284. request.method,
  1285. getRequestPathname(request)
  1286. );
  1287. }
  1288. return cookies;
  1289. }
  1290. };
  1291. __name(_CookiesParser, "CookiesParser");
  1292. var CookiesParser = _CookiesParser;
  1293. // src/parsers/request-parser.js
  1294. var import_http3 = require("http");
  1295. var import_js_format15 = require("@e22m4u/js-format");
  1296. var _RequestParser = class _RequestParser extends DebuggableService {
  1297. /**
  1298. * Parse.
  1299. *
  1300. * @param {IncomingMessage} request
  1301. * @returns {Promise<object>|object}
  1302. */
  1303. parse(request) {
  1304. if (!(request instanceof import_http3.IncomingMessage)) {
  1305. throw new import_js_format15.InvalidArgumentError(
  1306. "The first parameter of RequestParser.parse must be an instance of IncomingMessage, but %v was given.",
  1307. request
  1308. );
  1309. }
  1310. const data = {};
  1311. const promises = [];
  1312. const parsedQuery = this.getService(QueryParser).parse(request);
  1313. if (isPromise(parsedQuery)) {
  1314. promises.push(parsedQuery.then((v) => data.query = v));
  1315. } else {
  1316. data.query = parsedQuery;
  1317. }
  1318. const parsedCookies = this.getService(CookiesParser).parse(request);
  1319. if (isPromise(parsedCookies)) {
  1320. promises.push(parsedCookies.then((v) => data.cookies = v));
  1321. } else {
  1322. data.cookies = parsedCookies;
  1323. }
  1324. const parsedBody = this.getService(BodyParser).parse(request);
  1325. if (isPromise(parsedBody)) {
  1326. promises.push(parsedBody.then((v) => data.body = v));
  1327. } else {
  1328. data.body = parsedBody;
  1329. }
  1330. data.headers = Object.assign({}, request.headers);
  1331. return promises.length ? Promise.all(promises).then(() => data) : data;
  1332. }
  1333. };
  1334. __name(_RequestParser, "RequestParser");
  1335. var RequestParser = _RequestParser;
  1336. // src/route-registry.js
  1337. var import_js_path_trie = require("@e22m4u/js-path-trie");
  1338. var import_js_service2 = require("@e22m4u/js-service");
  1339. var import_js_format16 = require("@e22m4u/js-format");
  1340. var _RouteRegistry = class _RouteRegistry extends DebuggableService {
  1341. /**
  1342. * Constructor.
  1343. *
  1344. * @param {ServiceContainer} container
  1345. */
  1346. constructor(container) {
  1347. super(container);
  1348. this._trie = new import_js_path_trie.PathTrie();
  1349. }
  1350. /**
  1351. * Define route.
  1352. *
  1353. * @param {import('./route.js').RouteDefinition} routeDef
  1354. * @returns {Route}
  1355. */
  1356. defineRoute(routeDef) {
  1357. const debug = this.getDebuggerFor(this.defineRoute);
  1358. if (!routeDef || typeof routeDef !== "object" || Array.isArray(routeDef)) {
  1359. throw new import_js_format16.InvalidArgumentError(
  1360. "The route definition must be an Object, but %v was given.",
  1361. routeDef
  1362. );
  1363. }
  1364. const route = new Route(routeDef);
  1365. const triePath = `${route.method}/${route.path}`;
  1366. this._trie.add(triePath, route);
  1367. debug(
  1368. "The route %s %v was registered.",
  1369. route.method.toUpperCase(),
  1370. route.path
  1371. );
  1372. return route;
  1373. }
  1374. /**
  1375. * Match route by request.
  1376. *
  1377. * @param {import('http').IncomingRequest} request
  1378. * @returns {ResolvedRoute|undefined}
  1379. */
  1380. matchRouteByRequest(request) {
  1381. const debug = this.getDebuggerFor(this.matchRouteByRequest);
  1382. const requestPath = (request.url || "/").replace(/\?.*$/, "");
  1383. debug(
  1384. "Matching routes with the request %s %v.",
  1385. request.method.toUpperCase(),
  1386. requestPath
  1387. );
  1388. const rawTriePath = `${request.method.toUpperCase()}/${requestPath}`;
  1389. const triePath = rawTriePath.replace(/\/+/g, "/");
  1390. const resolved = this._trie.match(triePath);
  1391. if (resolved) {
  1392. const route = resolved.value;
  1393. debug(
  1394. "The route %s %v was matched.",
  1395. route.method.toUpperCase(),
  1396. route.path
  1397. );
  1398. const paramNames = Object.keys(resolved.params);
  1399. if (paramNames.length) {
  1400. paramNames.forEach((name) => {
  1401. debug(
  1402. "The path parameter %v had the value %v.",
  1403. name,
  1404. resolved.params[name]
  1405. );
  1406. });
  1407. } else {
  1408. debug("No path parameters found.");
  1409. }
  1410. return { route, params: resolved.params };
  1411. }
  1412. debug(
  1413. "No matched route for the request %s %v.",
  1414. request.method.toUpperCase(),
  1415. requestPath
  1416. );
  1417. }
  1418. };
  1419. __name(_RouteRegistry, "RouteRegistry");
  1420. var RouteRegistry = _RouteRegistry;
  1421. // src/request-context.js
  1422. var import_js_format17 = require("@e22m4u/js-format");
  1423. var import_js_service3 = require("@e22m4u/js-service");
  1424. var _RequestContext = class _RequestContext {
  1425. /**
  1426. * Service container.
  1427. *
  1428. * @type {ServiceContainer}
  1429. */
  1430. _container;
  1431. /**
  1432. * Getter of service container.
  1433. *
  1434. * @type {ServiceContainer}
  1435. */
  1436. get container() {
  1437. return this._container;
  1438. }
  1439. /**
  1440. * Request.
  1441. *
  1442. * @type {import('http').IncomingMessage}
  1443. */
  1444. _request;
  1445. /**
  1446. * Getter of request.
  1447. *
  1448. * @type {import('http').IncomingMessage}
  1449. */
  1450. get request() {
  1451. return this._request;
  1452. }
  1453. /**
  1454. * Response.
  1455. *
  1456. * @type {import('http').ServerResponse}
  1457. */
  1458. _response;
  1459. /**
  1460. * Getter of response.
  1461. *
  1462. * @type {import('http').ServerResponse}
  1463. */
  1464. get response() {
  1465. return this._response;
  1466. }
  1467. /**
  1468. * Route
  1469. *
  1470. * @type {Route}
  1471. */
  1472. _route;
  1473. /**
  1474. * Getter of route.
  1475. *
  1476. * @type {Route}
  1477. */
  1478. get route() {
  1479. return this._route;
  1480. }
  1481. /**
  1482. * Query.
  1483. *
  1484. * @type {object}
  1485. */
  1486. query = {};
  1487. /**
  1488. * Path parameters.
  1489. *
  1490. * @type {object}
  1491. */
  1492. params = {};
  1493. /**
  1494. * Headers.
  1495. *
  1496. * @type {object}
  1497. */
  1498. headers = {};
  1499. /**
  1500. * Parsed cookies.
  1501. *
  1502. * @type {object}
  1503. */
  1504. cookies = {};
  1505. /**
  1506. * Parsed body.
  1507. *
  1508. * @type {*}
  1509. */
  1510. body;
  1511. /**
  1512. * State.
  1513. *
  1514. * @type {object}
  1515. */
  1516. state = {};
  1517. /**
  1518. * Route meta.
  1519. *
  1520. * @type {import('./route.js').RouteMeta}
  1521. */
  1522. get meta() {
  1523. return this.route.meta;
  1524. }
  1525. /**
  1526. * Method.
  1527. *
  1528. * @returns {string}
  1529. */
  1530. get method() {
  1531. return this.request.method.toUpperCase();
  1532. }
  1533. /**
  1534. * Path.
  1535. *
  1536. * @returns {string}
  1537. */
  1538. get path() {
  1539. return this.request.url;
  1540. }
  1541. /**
  1542. * Pathname.
  1543. *
  1544. * @type {string|undefined}
  1545. * @private
  1546. */
  1547. _pathname = void 0;
  1548. /**
  1549. * Pathname.
  1550. *
  1551. * @returns {string}
  1552. */
  1553. get pathname() {
  1554. if (this._pathname != null) {
  1555. return this._pathname;
  1556. }
  1557. this._pathname = getRequestPathname(this.request);
  1558. return this._pathname;
  1559. }
  1560. /**
  1561. * Constructor.
  1562. *
  1563. * @param {ServiceContainer} container
  1564. * @param {import('http').IncomingMessage} request
  1565. * @param {import('http').ServerResponse} response
  1566. * @param {Route} route
  1567. */
  1568. constructor(container, request, response, route) {
  1569. if (!(0, import_js_service3.isServiceContainer)(container)) {
  1570. throw new import_js_format17.InvalidArgumentError(
  1571. 'The parameter "container" of RequestContext.constructor must be an instance of ServiceContainer, but %v was given.',
  1572. container
  1573. );
  1574. }
  1575. this._container = container;
  1576. if (!request || typeof request !== "object" || Array.isArray(request) || !isReadableStream(request)) {
  1577. throw new import_js_format17.InvalidArgumentError(
  1578. 'The parameter "request" of RequestContext.constructor must be an instance of IncomingMessage, but %v was given.',
  1579. request
  1580. );
  1581. }
  1582. this._request = request;
  1583. if (!response || typeof response !== "object" || Array.isArray(response) || !isWritableStream(response)) {
  1584. throw new import_js_format17.InvalidArgumentError(
  1585. 'The parameter "response" of RequestContext.constructor must be an instance of ServerResponse, but %v was given.',
  1586. response
  1587. );
  1588. }
  1589. this._response = response;
  1590. if (!(route instanceof Route)) {
  1591. throw new import_js_format17.InvalidArgumentError(
  1592. 'The parameter "route" of RequestContext.constructor must be an instance of Route, but %v was given.',
  1593. route
  1594. );
  1595. }
  1596. this._route = route;
  1597. }
  1598. };
  1599. __name(_RequestContext, "RequestContext");
  1600. var RequestContext = _RequestContext;
  1601. // src/trie-router.js
  1602. var import_js_service4 = require("@e22m4u/js-service");
  1603. var import_http4 = require("http");
  1604. // src/senders/data-sender.js
  1605. var import_js_format18 = require("@e22m4u/js-format");
  1606. var _DataSender = class _DataSender extends DebuggableService {
  1607. /**
  1608. * Send.
  1609. *
  1610. * @param {import('http').ServerResponse} response
  1611. * @param {*} data
  1612. * @returns {undefined}
  1613. */
  1614. send(response, data) {
  1615. const debug = this.getDebuggerFor(this.send);
  1616. if (data === response || response.headersSent) {
  1617. debug(
  1618. "Response sending was skipped because its headers where sent already."
  1619. );
  1620. return;
  1621. }
  1622. if (data == null) {
  1623. response.statusCode = 204;
  1624. response.end();
  1625. debug("The empty response was sent.");
  1626. return;
  1627. }
  1628. if (isReadableStream(data)) {
  1629. response.setHeader("Content-Type", "application/octet-stream");
  1630. data.pipe(response);
  1631. debug("The stream response was sent.");
  1632. return;
  1633. }
  1634. let debugMsg;
  1635. switch (typeof data) {
  1636. case "object":
  1637. case "boolean":
  1638. case "number":
  1639. if (Buffer.isBuffer(data)) {
  1640. response.setHeader("content-type", "application/octet-stream");
  1641. debugMsg = "The Buffer was sent as binary data.";
  1642. } else {
  1643. response.setHeader("content-type", "application/json");
  1644. debugMsg = (0, import_js_format18.format)("The %v was sent as JSON.", typeof data);
  1645. data = JSON.stringify(data);
  1646. }
  1647. break;
  1648. default:
  1649. response.setHeader("content-type", "text/plain");
  1650. debugMsg = "The response data was sent as plain text.";
  1651. data = String(data);
  1652. break;
  1653. }
  1654. response.end(data);
  1655. debug(debugMsg);
  1656. }
  1657. };
  1658. __name(_DataSender, "DataSender");
  1659. var DataSender = _DataSender;
  1660. // src/senders/error-sender.js
  1661. var import_util = require("util");
  1662. var import_statuses = __toESM(require("statuses"), 1);
  1663. var EXPOSED_ERROR_PROPERTIES = ["code", "details"];
  1664. var _ErrorSender = class _ErrorSender extends DebuggableService {
  1665. /**
  1666. * Handle.
  1667. *
  1668. * @param {import('http').IncomingMessage} request
  1669. * @param {import('http').ServerResponse} response
  1670. * @param {Error} error
  1671. * @returns {undefined}
  1672. */
  1673. send(request, response, error) {
  1674. const debug = this.getDebuggerFor(this.send);
  1675. let safeError = {};
  1676. if (error) {
  1677. if (typeof error === "object") {
  1678. safeError = error;
  1679. } else {
  1680. safeError = { message: String(error) };
  1681. }
  1682. }
  1683. const statusCode = error.statusCode || error.status || 500;
  1684. const body = { error: {} };
  1685. if (safeError.message && typeof safeError.message === "string") {
  1686. body.error.message = safeError.message;
  1687. } else {
  1688. body.error.message = (0, import_statuses.default)(statusCode);
  1689. }
  1690. EXPOSED_ERROR_PROPERTIES.forEach((name) => {
  1691. if (name in safeError) {
  1692. body.error[name] = safeError[name];
  1693. }
  1694. });
  1695. const requestData = {
  1696. url: request.url,
  1697. method: request.method,
  1698. headers: request.headers
  1699. };
  1700. const inspectOptions = {
  1701. showHidden: false,
  1702. depth: null,
  1703. colors: true,
  1704. compact: false
  1705. };
  1706. console.warn((0, import_util.inspect)(requestData, inspectOptions));
  1707. console.warn((0, import_util.inspect)(body, inspectOptions));
  1708. if (error.stack) {
  1709. console.log(error.stack);
  1710. } else {
  1711. console.error(error);
  1712. }
  1713. response.statusCode = statusCode;
  1714. response.setHeader("content-type", "application/json; charset=utf-8");
  1715. response.end(JSON.stringify(body, null, 2), "utf-8");
  1716. debug(
  1717. "The %s error was sent for the request %s %v.",
  1718. statusCode,
  1719. request.method,
  1720. getRequestPathname(request)
  1721. );
  1722. }
  1723. /**
  1724. * Send 404.
  1725. *
  1726. * @param {import('http').IncomingMessage} request
  1727. * @param {import('http').ServerResponse} response
  1728. * @returns {undefined}
  1729. */
  1730. send404(request, response) {
  1731. const debug = this.getDebuggerFor(this.send404);
  1732. response.statusCode = 404;
  1733. response.setHeader("content-type", "text/plain; charset=utf-8");
  1734. response.end("404 Not Found", "utf-8");
  1735. debug(
  1736. "The 404 error was sent for the request %s %v.",
  1737. request.method,
  1738. getRequestPathname(request)
  1739. );
  1740. }
  1741. };
  1742. __name(_ErrorSender, "ErrorSender");
  1743. var ErrorSender = _ErrorSender;
  1744. // src/trie-router.js
  1745. var _TrieRouter = class _TrieRouter extends DebuggableService {
  1746. /**
  1747. * Define route.
  1748. *
  1749. * Example 1:
  1750. * ```
  1751. * const router = new TrieRouter();
  1752. * router.defineRoute({
  1753. * method: HttpMethod.GET, // Request method.
  1754. * path: '/', // Path template.
  1755. * handler: ctx => 'Hello world!', // Request handler.
  1756. * });
  1757. * ```
  1758. *
  1759. * Example 2:
  1760. * ```
  1761. * const router = new TrieRouter();
  1762. * router.defineRoute({
  1763. * method: HttpMethod.POST, // Request method.
  1764. * path: '/users/:id', // The path template may have parameters.
  1765. * preHandler(ctx) { ... }, // The "preHandler" executes before a route handler.
  1766. * handler(ctx) { ... }, // Request handler function.
  1767. * postHandler(ctx, data) { ... }, // The "postHandler" executes after a route handler.
  1768. * });
  1769. * ```
  1770. *
  1771. * @param {import('./route-registry.js').RouteDefinition} routeDef
  1772. * @returns {import('./route.js').Route}
  1773. */
  1774. defineRoute(routeDef) {
  1775. return this.getService(RouteRegistry).defineRoute(routeDef);
  1776. }
  1777. /**
  1778. * Request listener.
  1779. *
  1780. * Example:
  1781. * ```
  1782. * import http from 'http';
  1783. * import {TrieRouter} from '@e22m4u/js-trie-router';
  1784. *
  1785. * const router = new TrieRouter();
  1786. * const server = new http.Server();
  1787. * server.on('request', router.requestListener); // Sets the request listener.
  1788. * server.listen(3000); // Starts listening for connections.
  1789. * ```
  1790. *
  1791. * @returns {Function}
  1792. */
  1793. get requestListener() {
  1794. return this._handleRequest.bind(this);
  1795. }
  1796. /**
  1797. * Handle incoming request.
  1798. *
  1799. * @param {import('http').IncomingMessage} request
  1800. * @param {import('http').ServerResponse} response
  1801. * @returns {Promise<undefined>}
  1802. * @private
  1803. */
  1804. async _handleRequest(request, response) {
  1805. const debug = this.getDebuggerFor(this._handleRequest);
  1806. const requestPath = (request.url || "/").replace(/\?.*$/, "");
  1807. debug(
  1808. "Preparing to handle an incoming request %s %v.",
  1809. request.method,
  1810. requestPath
  1811. );
  1812. const resolved = this.getService(RouteRegistry).matchRouteByRequest(request);
  1813. if (!resolved) {
  1814. debug("No route for the request %s %v.", request.method, requestPath);
  1815. this.getService(ErrorSender).send404(request, response);
  1816. } else {
  1817. const { route, params } = resolved;
  1818. const container = new import_js_service4.ServiceContainer(this.container);
  1819. const context = new RequestContext(container, request, response, route);
  1820. container.set(RequestContext, context);
  1821. container.set(import_http4.IncomingMessage, request);
  1822. container.set(import_http4.ServerResponse, response);
  1823. context.params = params;
  1824. let data;
  1825. try {
  1826. const reqDataOrPromise = this.getService(RequestParser).parse(request);
  1827. if (isPromise(reqDataOrPromise)) {
  1828. const reqData = await reqDataOrPromise;
  1829. Object.assign(context, reqData);
  1830. } else {
  1831. Object.assign(context, reqDataOrPromise);
  1832. }
  1833. const hookInvoker = this.getService(HookInvoker);
  1834. data = hookInvoker.invokeAndContinueUntilValueReceived(
  1835. route,
  1836. RouterHookType.PRE_HANDLER,
  1837. response,
  1838. context
  1839. );
  1840. if (isPromise(data)) {
  1841. data = await data;
  1842. }
  1843. if (!isResponseSent(response) && data == null) {
  1844. data = route.handle(context);
  1845. if (isPromise(data)) {
  1846. data = await data;
  1847. }
  1848. let postHandlerData = hookInvoker.invokeAndContinueUntilValueReceived(
  1849. route,
  1850. RouterHookType.POST_HANDLER,
  1851. response,
  1852. context,
  1853. data
  1854. );
  1855. if (isPromise(postHandlerData)) {
  1856. postHandlerData = await postHandlerData;
  1857. }
  1858. if (postHandlerData != null) {
  1859. data = postHandlerData;
  1860. }
  1861. }
  1862. } catch (error) {
  1863. this.getService(ErrorSender).send(request, response, error);
  1864. return;
  1865. }
  1866. if (!isResponseSent(response)) {
  1867. this.getService(DataSender).send(response, data);
  1868. }
  1869. }
  1870. }
  1871. /**
  1872. * Add hook.
  1873. *
  1874. * Example:
  1875. * ```
  1876. * import {TrieRouter} from '@e22m4u/js-trie-router';
  1877. * import {RouterHookType} from '@e22m4u/js-trie-router';
  1878. *
  1879. * // Router instance.
  1880. * const router = new TrieRouter();
  1881. *
  1882. * // Adds the "preHandler" hook for each route.
  1883. * router.addHook(
  1884. * RouterHookType.PRE_HANDLER,
  1885. * ctx => { ... },
  1886. * );
  1887. *
  1888. * // Adds the "postHandler" hook for each route.
  1889. * router.addHook(
  1890. * RouterHookType.POST_HANDLER,
  1891. * ctx => { ... },
  1892. * );
  1893. * ```
  1894. *
  1895. * @param {string} type
  1896. * @param {Function} hook
  1897. * @returns {this}
  1898. */
  1899. addHook(type, hook) {
  1900. this.getService(HookRegistry).addHook(type, hook);
  1901. return this;
  1902. }
  1903. /**
  1904. * Add pre-handler hook.
  1905. *
  1906. * @param {Function} hook
  1907. * @returns {this}
  1908. */
  1909. addPreHandler(hook) {
  1910. this.getService(HookRegistry).addHook(RouterHookType.PRE_HANDLER, hook);
  1911. return this;
  1912. }
  1913. /**
  1914. * Add post-handler hook.
  1915. *
  1916. * @param {Function} hook
  1917. * @returns {this}
  1918. */
  1919. addPostHandler(hook) {
  1920. this.getService(HookRegistry).addHook(RouterHookType.POST_HANDLER, hook);
  1921. return this;
  1922. }
  1923. };
  1924. __name(_TrieRouter, "TrieRouter");
  1925. var TrieRouter = _TrieRouter;
  1926. // Annotate the CommonJS export names for ESM import in node:
  1927. 0 && (module.exports = {
  1928. BodyParser,
  1929. CHARACTER_ENCODING_LIST,
  1930. CookiesParser,
  1931. DataSender,
  1932. EXPOSED_ERROR_PROPERTIES,
  1933. ErrorSender,
  1934. HookInvoker,
  1935. HookRegistry,
  1936. HttpMethod,
  1937. METHODS_WITH_BODY,
  1938. QueryParser,
  1939. RequestContext,
  1940. RequestParser,
  1941. Route,
  1942. RouteRegistry,
  1943. RouterHookType,
  1944. RouterOptions,
  1945. TrieRouter,
  1946. UNPARSABLE_MEDIA_TYPES,
  1947. cloneDeep,
  1948. createCookieString,
  1949. createError,
  1950. createRequestMock,
  1951. createResponseMock,
  1952. createRouteMock,
  1953. fetchRequestBody,
  1954. getRequestPathname,
  1955. isPromise,
  1956. isReadableStream,
  1957. isResponseSent,
  1958. isWritableStream,
  1959. parseContentType,
  1960. parseCookieString,
  1961. parseJsonBody,
  1962. toCamelCase
  1963. });