@visulima/api-platform 1.0.2 → 1.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +23 -0
- package/bin/index.js +46 -0
- package/dist/chunk-AOL5OFCG.mjs +284 -0
- package/dist/chunk-AOL5OFCG.mjs.map +1 -0
- package/dist/chunk-ATZDRT65.js +284 -0
- package/dist/chunk-ATZDRT65.js.map +1 -0
- package/dist/{chunk-AJKZCWFG.js → chunk-F7RHRCUQ.js} +2 -2
- package/dist/chunk-F7RHRCUQ.js.map +1 -0
- package/dist/{chunk-XXZ56SKG.mjs → chunk-RPHC5ZGB.mjs} +2 -2
- package/dist/chunk-RPHC5ZGB.mjs.map +1 -0
- package/dist/index-server.d.ts +4 -14
- package/dist/index-server.js +9 -10
- package/dist/index-server.js.map +1 -1
- package/dist/index-server.mjs +7 -8
- package/dist/index-server.mjs.map +1 -1
- package/dist/next/cli/index.js +8 -8
- package/dist/next/cli/index.js.map +1 -1
- package/dist/next/cli/index.mjs +8 -8
- package/dist/next/cli/index.mjs.map +1 -1
- package/dist/next/index-browser.js +2 -2
- package/dist/next/index-browser.mjs +1 -1
- package/dist/next/index-server.d.ts +14 -13
- package/dist/next/index-server.js +4 -4
- package/dist/next/index-server.js.map +1 -1
- package/dist/next/index-server.mjs +2 -2
- package/dist/next/index-server.mjs.map +1 -1
- package/dist/swagger-handler-ffed72c2.d.ts +19 -0
- package/next/cli/package.json +12 -3
- package/next/package.json +9 -0
- package/package.json +20 -20
- package/recipes/api/swagger.ts +12 -0
- package/recipes/pages/redoc-ui.tsx +5 -0
- package/recipes/pages/swagger-ui.tsx +5 -0
- package/dist/chunk-2LATTLUM.mjs +0 -166
- package/dist/chunk-2LATTLUM.mjs.map +0 -1
- package/dist/chunk-AJKZCWFG.js.map +0 -1
- package/dist/chunk-S7GUPAL4.js +0 -166
- package/dist/chunk-S7GUPAL4.js.map +0 -1
- package/dist/chunk-XXZ56SKG.mjs.map +0 -1
- package/src/connect/create-node-router.ts +0 -44
- package/src/connect/handler.ts +0 -46
- package/src/connect/middleware/cors-middleware.ts +0 -10
- package/src/connect/middleware/http-header-normalizer.ts +0 -93
- package/src/connect/middleware/rate-limiter-middleware.ts +0 -43
- package/src/connect/middleware/serializers-middleware.ts +0 -121
- package/src/connect/serializers/types.d.ts +0 -1
- package/src/connect/serializers/xml.ts +0 -13
- package/src/connect/serializers/yaml.ts +0 -7
- package/src/error-handler/jsonapi-error-handler.ts +0 -46
- package/src/error-handler/problem-error-handler.ts +0 -44
- package/src/error-handler/types.d.ts +0 -14
- package/src/error-handler/utils.ts +0 -39
- package/src/index-browser.tsx +0 -1
- package/src/index-server.ts +0 -75
- package/src/next/cli/index.ts +0 -2
- package/src/next/cli/list/api-route-file-parser.ts +0 -74
- package/src/next/cli/list/collect-api-route-files.ts +0 -42
- package/src/next/cli/list/list-command.ts +0 -105
- package/src/next/cli/list/routes-render.ts +0 -62
- package/src/next/cli/list/types.d.ts +0 -1
- package/src/next/index-browser.tsx +0 -3
- package/src/next/index-server.ts +0 -6
- package/src/next/routes/api/swagger.ts +0 -23
- package/src/next/routes/pages/swagger/get-static-properties-swagger.ts +0 -32
- package/src/next/routes/pages/swagger/redoc.tsx +0 -35
- package/src/next/routes/pages/swagger/swagger.tsx +0 -44
- package/src/next/webpack/with-open-api.ts +0 -63
- package/src/swagger/extend-swagger-spec.ts +0 -167
- package/src/swagger/swagger-handler.ts +0 -83
- package/src/utils.ts +0 -37
- package/src/zod/date-in-schema.ts +0 -57
- package/src/zod/date-out-schema.ts +0 -41
- package/src/zod/index.ts +0 -9
package/dist/index-server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"names":["default","HttpError","getReasonPhrase","defaultTitle","expressWrapper","NodeRouter","sendJson","createHttpError"],"mappings":";;;;;;;;;;;;AAEA;AAAA,EACe,WAAXA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACG;;;AC7CP,SAAS,kBAAkB;;;ACG3B,OAAO,qBAAqB;;;ACH5B,SAAS,iBAAiB;AAC1B,SAAS,uBAAuB;AAChC,SAAS,iBAAiB,iBAAiB;;;ACF3C,SAAS,mBAAmB;AAGrB,IAAM,kBAAkB,CAAC,UAA0B,UAAe;AACrE,QAAM,UAAsE,MAAM,WAAW,CAAC;AAE9F,SAAO,KAAK,OAAO,EAAE,QAAQ,CAAC,WAAmB;AAC7C,aAAS,UAAU,QAAQ,QAAQ,OAAkD;AAAA,EACzF,CAAC;AACL;AAOO,IAAM,WAAW,CAAC,UAA0B,aAAwB;AAEvE,WAAS,UAAU,gBAAgB,iCAAiC;AAEpE,WAAS,IAAI,KAAK,UAAU,QAAQ,CAAC;AACzC;AAEO,IAAM,0BAA0B,CAAC,UAA0B,UAAqB;AAEnF,MAAI,OAAO,MAAM,eAAe,aAAa;AACzC,aAAS,aAAa,MAAM;AAAA,EAChC;AAGA,MAAI,OAAO,MAAM,WAAW,aAAa;AACrC,aAAS,aAAa,MAAM;AAAA,EAChC;AAGA,MAAI,SAAS,aAAa,KAAK;AAC3B,aAAS,aAAa,YAAY;AAAA,EACtC;AACJ;;;AD/BA,IAAM,eAAe;AAErB,IAAM,sBAAoC,CAAC,OAAsC,UAAU,aAAa;AACpG,0BAAwB,UAAU,KAAK;AAEvC,kBAAgB,UAAU,KAAK;AAE/B,MAAI,iBAAiB,aAAa,UAAU,gBAAgB,KAAK,GAAG;AAChE,UAAM,aAAa,IAAI,gBAAgB;AAEvC,aAAS,UAAU,WAAW,UAAU,KAAK,CAAC;AAAA,EAClD,WAAW,iBAAiB,WAAW;AACnC,UAAM,EAAE,YAAY,OAAO,QAAQ,IAAI;AAEvC,aAAS,UAAU;AAAA,MACf,QAAQ;AAAA,QACJ;AAAA,UACI,MAAM;AAAA,UACN,OAAO,SAAS,gBAAgB,UAAU,KAAK;AAAA,UAC/C,QAAQ;AAAA,QACZ;AAAA,MACJ;AAAA,IACJ,CAAC;AAAA,EACL,OAAO;AACH,UAAM,EAAE,QAAQ,IAAI;AAEpB,aAAS,UAAU;AAAA,MACf,QAAQ;AAAA,QACJ;AAAA,UACI,MAAM;AAAA,UACN,OAAO,gBAAgB,SAAS,UAAU,KAAK;AAAA,UAC/C,QAAQ;AAAA,QACZ;AAAA,MACJ;AAAA,IACJ,CAAC;AAAA,EACL;AACJ;AAEA,IAAO,gCAAQ;;;AE7Cf,SAAS,aAAAC,kBAAiB;AAC1B,SAAS,mBAAAC,wBAAuB;AAKhC,IAAM,cAAc;AACpB,IAAMC,gBAAe;AAMrB,IAAM,sBAAoC,CAAC,OAA0B,UAAU,aAAa;AACxF,QAAM,EAAE,OAAO,QAAQ,IAAI;AAE3B,MAAI,iBAAiBF,YAAW;AAC5B,UAAM;AAAA,MACF;AAAA,MAAY;AAAA,MAAQ;AAAA,MAAO;AAAA,IAC/B,IAAI;AAEJ,aAAS,aAAa;AAEtB,oBAAgB,UAAU,KAAK;AAE/B,aAAS,UAAU;AAAA,MACf,MAAM,QAAQ;AAAA,MACd,OAAO,SAASC,iBAAgB,UAAU,KAAKC;AAAA,MAC/C,SAAS;AAAA,MACT,GAAI,SAAS,EAAE,OAAO,MAAM,IAAI,CAAC;AAAA,IACrC,CAAC;AAAA,EACL,OAAO;AACH,4BAAwB,UAAU,KAAK;AAEvC,aAAS,UAAU;AAAA,MACf,MAAM;AAAA,MACN,OAAOD,iBAAgB,SAAS,UAAU,KAAKC;AAAA,MAC/C,SAAS;AAAA,MACT,GAAK,MAAsC,SAAS,EAAE,OAAO,MAAM,IAAI,CAAC;AAAA,IAC5E,CAAC;AAAA,EACL;AACJ;AAEA,IAAO,gCAAQ;;;AHhCR,IAAM,UAAU,CAAmE,eAA8B,cAAuB,OAAO,OAAgB,SAAkB,aAAsC;AAC1N,QAAM,YAAoB,QAAQ,QAAQ;AAE1C,MAAI,eAA6B;AAEjC,MAAI,cAAc,4BAA4B;AAC1C,mBAAe;AAAA,EACnB;AAGA,aAAW,EAAE,OAAO,QAAQ,KAAK,eAAe;AAC5C,QAAI,MAAM,KAAK,SAAS,GAAG;AACvB,qBAAe;AACf;AAAA,IACJ;AAAA,EACJ;AAGA,EAAC,MAAsC,SAAS;AAEhD,eAAa,OAAO,SAAS,QAAQ;AACzC;AAEO,IAAM,YAIe,OAAO,SAAS,UAAU,WAAW;AAC7D,QAAM,gBAAgB,CAAC,GAAG,IAAI,IAAI,OAAO,IAAI,CAAC,UAAU,MAAM,MAAM,CAAC,CAAC,EAAE,KAAK,IAAI;AAEjF,WAAS,UAAU,SAAS,aAAa;AACzC,WAAS,aAAa;AAEtB,QAAM,gBAAgB,KAAK,kBAAkB,QAAQ,uBAAuB;AAChF;;;AI1CA,IAAM,aAAa;AAAA,EACf,MAAM;AAAA,EACN,SAAS;AAAA,EACT,cAAc;AAAA,EACd,oBAAoB;AAAA,EACpB,cAAc;AAAA,EACd,eAAe;AAAA,EACf,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AAAA,EACL,MAAM;AAAA,EACN,YAAY;AAAA,EACZ,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,gBAAgB;AAAA,EAChB,6BAA6B;AAAA,EAC7B,wBAAwB;AAAA,EACxB,4BAA4B;AAAA,EAC5B,qBAAqB;AAAA,EACrB,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,MAAM;AAAA,EACN,KAAK;AAAA,EACL,IAAI;AAAA,EACJ,KAAK;AAAA,EACL,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,yBAAyB;AAAA,EACzB,UAAU;AACd;AAEA,IAAM,qBAAqB,CAAC,KAAa,cAAuB;AAC5D,QAAM,eAAe,IAAI,YAAY;AAErC,MAAI,CAAC,WAAW;AACZ,WAAO;AAAA,EACX;AAEA,MAAI,WAAW,eAA0C;AACrD,WAAO,WAAW;AAAA,EACtB;AAEA,SACI,aACK,MAAM,GAAG,EAET,IAAI,CAAC,SAAM;AAjDxB;AAiD2B,uBAAK,OAAL,mBAAS,iBAAgB,KAAK,MAAM,CAAC;AAAA,GAAC,EACpD,KAAK,GAAG;AAErB;AAEA,IAAM,WAAW;AAAA,EACb,WAAW;AAAA,EACX;AACJ;AAOA,IAAM,iCAAiC,CAAC,aAAyG;AAC7I,QAAM,UAAU,EAAE,GAAG,UAAU,GAAG,SAAS;AAE3C,SAAO,OAAwC,SAAkB,GAAQ,SAAsB;AAC3F,QAAI,QAAQ,SAAS;AACjB,YAAM,aAAkC,CAAC;AACzC,YAAM,UAA+B,CAAC;AAEtC,aAAO,KAAK,QAAQ,OAAO,EAAE,QAAQ,CAAC,QAAQ;AAC1C,mBAAW,OAAO,QAAQ,QAAQ;AAElC,cAAM,gBAAgB,QAAQ,mBAAmB,KAAK,QAAQ,SAAS;AAEvE,YAAI,OAAO,kBAAkB,aAAa;AACtC,kBAAQ,iBAAiB,QAAQ,QAAQ;AAAA,QAC7C;AAAA,MACJ,CAAC;AAED,cAAQ,UAAU;AAGlB,cAAQ,aAAa;AAAA,IACzB;AAEA,WAAO,KAAK;AAAA,EAChB;AACJ;AAEA,IAAO,iCAAQ;;;AC3Ff,OAAO,aAAa;AACpB,SAAS,UAAU,kBAAkB;;;ACDrC,SAAS,aAAa;AAItB,IAAM,SAAS;AAAA,EACX,QAAQ;AAAA,EACR,QAAQ;AACZ;AAEA,IAAM,iBAA6B,CAAC,SAAqC,MAAM,MAAM,MAAM;AAE3F,IAAO,cAAQ;;;ADFf,SAAS,iBAAiB,SAAuB;AAC7C,MAAI,OAAO,YAAY,UAAU;AAC7B,WAAO;AAAA,EACX;AAEA,MAAI;AACA,UAAM,SAAS,KAAK,MAAM,OAAO;AACjC,UAAM,OAAO,OAAO,UAAU,SAAS,KAAK,MAAM;AAElD,WAAO,SAAS,qBAAqB,SAAS;AAAA,EAClD,QAAE;AACE,WAAO;AAAA,EACX;AACJ;AAEA,IAAM,iBAAiB;AAGvB,IAAM,YAAY,CACd,aACA,SACA,UACA,MACA,YAIC;AACD,QAAM,cAAc,SAAS,UAAU,cAAc;AAGrD,MAAI,OAAO,gBAAgB,UAAU;AACjC,WAAO;AAAA,EACX;AAEA,QAAM,SAAS,QAAQ,OAAO;AAC9B,QAAM,QAAkB,CAAC,GAAI,OAAO,MAAM,GAAgB,QAAQ,kBAAkB;AAEpF,MAAI,iBAAiB;AAGrB,QAAM,MAAM,CAAC,SAAS;AAnD1B;AAoDQ,QAAI,aAAa;AAEjB,gBAAY,QAAQ,CAAC,EAAE,OAAO,WAAW,MAAM;AAC3C,UAAI,CAAC,MAAM,KAAK,IAAI,GAAG;AACnB;AAAA,MACJ;AAEA,eAAS,UAAU,gBAAgB,IAAI;AACvC,uBAAiB,WAAW,cAAc;AAC1C,mBAAa;AAAA,IACjB,CAAC;AAED,QAAI,CAAC,YAAY;AACb,UAAI,WAAW,KAAK,IAAI,GAAG;AACvB,iBAAS,UAAU,gBAAgB,IAAI;AAEvC,yBAAiB,aAAgB,iBAAiB,IAAI,IAAI,KAAK,MAAM,IAAI,IAAI,IAAI;AAAA,MACrF,WAAW,MAAM,KAAK,IAAI,GAAG;AACzB,iBAAS,UAAU,gBAAgB,IAAI;AAEvC,yBAAiB,YAAe;AAAA,UAC5B,CAAC,WAAW,IAAG,aAAQ,QAAR,mBAAa,QAAQ,SAAS,MAAM,KAAK,CAAC,IAAI,iBAAiB,IAAI,IAAI,KAAK,MAAM,IAAI,IAAI;AAAA,QAC7G,CAAC;AAAA,MACL;AAAA,IACJ;AAEA,WAAO;AAAA,EACX,CAAC;AAGD,SAAO;AACX;AAGA,IAAM,wBAAwB,CAAC,cAA2B,CAAC,GAAG,qBAA6B,sCAAsC,OAAyE,SAAkB,UAAsC,SAAsB;AACpR,MAAI,QAAQ,qCAA8B,UAAS,YAAY;AAC3D,UAAM,UAAW,SAA6B;AAE9C,IAAC,SAA6B,OAAO,CAAC,SAAS;AAC3C,MAAC,SAA6B,OAAO;AAGrC,aAAO,UAA6B,aAAa,SAAS,UAAU,MAAM,EAAE,mBAAmB,CAAC;AAEhG,aAAQ,SAA6B,KAAK,IAAI;AAAA,IAClD;AAAA,EACJ,OAAO;AACH,UAAM,SAAS,SAAS;AAGxB,aAAS,MAAM,CAAC,SAAS,eAAe;AACpC,eAAS,MAAM;AAGf,aAAO,UAA6B,aAAa,SAAS,UAAU,MAAM,EAAE,mBAAmB,CAAC;AAGhG,aAAO,SAAS,IAAI,MAAM,GAAG,UAAU;AAAA,IAC3C;AAAA,EACJ;AAEA,SAAO,KAAK;AAChB;AAMA,IAAO,iCAAQ;;;AN7Gf,IAAM,mBAAmB,CAKjB,UAUI,CAAC,MACJ;AA3BT;AA4BI,QAAM,SAAS,IAAI,WAAsC;AAAA,IACrD;AAAA,IACA,SAAS,QAAQ,QAAQ,iBAAiB,CAAC,GAAG,QAAQ,aAAa,KAAK;AAAA,EAC5E,CAAC;AAED,SAAO,OACF,IAAI,iCAA+B,wCAAS,gBAAT,mBAAuB,8BAA6B,CAAC,CAAC,CAAC,EAC1F;AAAA,IACG;AAAA,QACI,8CAAS,gBAAT,mBAAsB,gBAAtB,mBAAmC,gBAAe,CAAC;AAAA,QACnD,8CAAS,gBAAT,mBAAsB,gBAAtB,mBAAmC,uBAAsB;AAAA,IAC7D;AAAA,EACJ;AACR;AAEA,IAAO,6BAAQ;;;ADoBf;AAAA,EACI;AAAA,EAAkB;AAAA,EAAY,kBAAAC;AAAA,EAAgB,cAAAC;AAAA,EAAY;AAAA,EAAQ;AAAA,EAAS,YAAAC;AAAA,OACxE;;;AShEP,OAAOC,sBAAqB;AAM5B,IAAM,QAA4E,CAAC,aAAY,mCAAS,OAChG,QAAQ,QAAQ,sBAChB,QAAQ,QAAQ,gBACjB,QAAQ,WAAW;AAK1B,IAAM,wBAAwB,CAAC,aAAkC,YAAkF,OAAyE,SAAkB,UAAsC,SAAsB;AACtS,QAAM,KAAK,MAAM,OAAO;AAExB,MAAI,OAAO,OAAO,aAAa;AAC3B,UAAMA,iBAAgB,KAAK,YAAY;AAAA,EAC3C;AAEA,MAAI;AACA,UAAM,UAAU,MAAM,YAAY,QAAQ,EAAE;AAE5C,UAAM,gBAAgD;AAAA,MAClD,eAAe,KAAK,MAAM,QAAQ,eAAe,GAAI,KAAK;AAAA,MAC1D,yBAAyB,QAAQ;AAAA,MACjC,qBAAqB,IAAI,KAAK,KAAK,IAAI,IAAI,QAAQ,YAAY,EAAE,YAAY;AAAA,MAC7E,GAAG;AAAA,IACP;AAEA,WAAO,KAAK,aAAa,EAAE,QAAQ,CAAC,QAAQ;AACxC,eAAS,UAAU,KAAK,cAAc,IAAmB;AAAA,IAC7D,CAAC;AAED,UAAM,KAAK;AAAA,EACf,QAAE;AACE,UAAMA,iBAAgB,KAAK,mBAAmB;AAAA,EAClD;AACJ;AAEA,IAAO,kCAAQ;;;AC1Cf,SAAS,sBAAsB;AAG/B,OAAO,UAAU;AAIjB,IAAM,iBAAiB,CAAmE,YAAgD,eAAkC,KAAK,OAAO,CAAC;AAEzL,IAAO,0BAAQ","sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (typeof error.statusCode !== \"undefined\") {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (typeof error.status !== \"undefined\") {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (typeof normalizedKey !== \"undefined\") {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst config = {\n header: true,\n indent: \" \",\n};\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, config);\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (typeof ip === \"undefined\") {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"names":["default","HttpError","getReasonPhrase","defaultTitle","expressWrapper","NodeRouter","sendJson","createHttpError"],"mappings":"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","sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (error.statusCode !== undefined) {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (error.status !== undefined) {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (normalizedKey !== undefined) {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, {\n header: true,\n indent: \" \",\n});\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (ip === undefined) {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"]}
|
package/dist/index-server.mjs
CHANGED
|
@@ -6,7 +6,7 @@ import {
|
|
|
6
6
|
import {
|
|
7
7
|
swagger_handler_default,
|
|
8
8
|
yaml_default
|
|
9
|
-
} from "./chunk-
|
|
9
|
+
} from "./chunk-AOL5OFCG.mjs";
|
|
10
10
|
import "./chunk-5VRACIDE.mjs";
|
|
11
11
|
|
|
12
12
|
// src/index-server.ts
|
|
@@ -79,10 +79,10 @@ var sendJson = (response, jsonBody) => {
|
|
|
79
79
|
response.end(JSON.stringify(jsonBody));
|
|
80
80
|
};
|
|
81
81
|
var addStatusCodeToResponse = (response, error) => {
|
|
82
|
-
if (
|
|
82
|
+
if (error.statusCode !== void 0) {
|
|
83
83
|
response.statusCode = error.statusCode;
|
|
84
84
|
}
|
|
85
|
-
if (
|
|
85
|
+
if (error.status !== void 0) {
|
|
86
86
|
response.statusCode = error.status;
|
|
87
87
|
}
|
|
88
88
|
if (response.statusCode < 400) {
|
|
@@ -238,7 +238,7 @@ var httpHeaderNormalizerMiddleware = (options_) => {
|
|
|
238
238
|
Object.keys(request.headers).forEach((key) => {
|
|
239
239
|
rawHeaders[key] = request.headers[key];
|
|
240
240
|
const normalizedKey = options.normalizeHeaderKey(key, options.canonical);
|
|
241
|
-
if (
|
|
241
|
+
if (normalizedKey !== void 0) {
|
|
242
242
|
headers[normalizedKey] = request.headers[key];
|
|
243
243
|
}
|
|
244
244
|
});
|
|
@@ -256,11 +256,10 @@ import { header as headerCase } from "case";
|
|
|
256
256
|
|
|
257
257
|
// src/connect/serializers/xml.ts
|
|
258
258
|
import { toXML } from "jstoxml";
|
|
259
|
-
var
|
|
259
|
+
var xmlTransformer = (data) => toXML(data, {
|
|
260
260
|
header: true,
|
|
261
261
|
indent: " "
|
|
262
|
-
};
|
|
263
|
-
var xmlTransformer = (data) => toXML(data, config);
|
|
262
|
+
});
|
|
264
263
|
var xml_default = xmlTransformer;
|
|
265
264
|
|
|
266
265
|
// src/connect/middleware/serializers-middleware.ts
|
|
@@ -363,7 +362,7 @@ import createHttpError2 from "http-errors";
|
|
|
363
362
|
var getIP = (request) => (request == null ? void 0 : request.ip) || request.headers["x-forwarded-for"] || request.headers["x-real-ip"] || request.connection.remoteAddress;
|
|
364
363
|
var rateLimiterMiddleware = (rateLimiter, headers) => async (request, response, next) => {
|
|
365
364
|
const ip = getIP(request);
|
|
366
|
-
if (
|
|
365
|
+
if (ip === void 0) {
|
|
367
366
|
throw createHttpError2(400, "Missing IP");
|
|
368
367
|
}
|
|
369
368
|
try {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (typeof error.statusCode !== \"undefined\") {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (typeof error.status !== \"undefined\") {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (typeof normalizedKey !== \"undefined\") {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst config = {\n header: true,\n indent: \" \",\n};\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, config);\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (typeof ip === \"undefined\") {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"],"mappings":"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","names":["default","HttpError","getReasonPhrase","defaultTitle","HttpError","getReasonPhrase","expressWrapper","NodeRouter","sendJson","createHttpError"]}
|
|
1
|
+
{"version":3,"sources":["../src/index-server.ts","../src/connect/create-node-router.ts","../src/connect/handler.ts","../src/error-handler/jsonapi-error-handler.ts","../src/error-handler/utils.ts","../src/error-handler/problem-error-handler.ts","../src/connect/middleware/http-header-normalizer.ts","../src/connect/middleware/serializers-middleware.ts","../src/connect/serializers/xml.ts","../src/connect/middleware/rate-limiter-middleware.ts","../src/connect/middleware/cors-middleware.ts"],"sourcesContent":["export * from \"./index-browser\";\n\nexport {\n default as createHttpError,\n BadRequest,\n Forbidden,\n BadGateway,\n BandwidthLimitExceeded,\n Conflict,\n ExpectationFailed,\n FailedDependency,\n GatewayTimeout,\n Gone,\n HTTPVersionNotSupported,\n ImATeapot,\n InsufficientStorage,\n InternalServerError,\n VariantAlsoNegotiates,\n ProxyAuthenticationRequired,\n NetworkAuthenticationRequire,\n LengthRequired,\n LoopDetected,\n Locked,\n MethodNotAllowed,\n MisdirectedRequest,\n NotAcceptable,\n NotExtended,\n NotFound,\n NotImplemented,\n PayloadTooLarge,\n RequestHeaderFieldsTooLarge,\n PaymentRequired,\n PreconditionFailed,\n PreconditionRequired,\n RangeNotSatisfiable,\n RequestTimeout,\n ServiceUnavailable,\n TooManyRequests,\n Unauthorized,\n UnorderedCollection,\n UnprocessableEntity,\n UnavailableForLegalReasons,\n UnsupportedMediaType,\n UpgradeRequired,\n URITooLong,\n} from \"http-errors\";\n\nexport { default as createNodeRouter } from \"./connect/create-node-router\";\nexport { onError, onNoMatch } from \"./connect/handler\";\nexport type {\n EdgeRequestHandler,\n ExpressRequestHandler,\n NodeRequestHandler,\n Route,\n HandlerOptions,\n NextHandler,\n FunctionLike,\n Nextable,\n ValueOrPromise,\n FindResult,\n RouteShortcutMethod,\n HttpMethod,\n} from \"@visulima/connect\";\nexport {\n createEdgeRouter, EdgeRouter, expressWrapper, NodeRouter, Router, withZod, sendJson,\n} from \"@visulima/connect\";\n\nexport { default as rateLimiterMiddleware } from \"./connect/middleware/rate-limiter-middleware\";\nexport { default as corsMiddleware } from \"./connect/middleware/cors-middleware\";\nexport { default as serializersMiddleware } from \"./connect/middleware/serializers-middleware\";\nexport { default as httpHeaderNormalizerMiddleware } from \"./connect/middleware/http-header-normalizer\";\n\nexport { default as swaggerHandler } from \"./swagger/swagger-handler\";\n\nexport { dateIn, dateOut } from \"./zod\";\n","import { NodeRouter } from \"@visulima/connect\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { AnyZodObject } from \"zod\";\nimport { ZodObject } from \"zod\";\n\nimport type { ErrorHandlers } from \"../error-handler/types\";\nimport { onError, onNoMatch } from \"./handler\";\nimport httpHeaderNormalizerMiddleware from \"./middleware/http-header-normalizer\";\nimport type { Serializers } from \"./middleware/serializers-middleware\";\nimport serializersMiddleware from \"./middleware/serializers-middleware\";\n\nconst createNodeRouter = <\n Request extends IncomingMessage,\n Response extends ServerResponse,\n Schema extends AnyZodObject = ZodObject<{ body?: AnyZodObject; headers?: AnyZodObject; query?: AnyZodObject }>,\n>(\n options: {\n middlewares?: {\n \"http-header-normalizer\"?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string };\n serializers?: {\n serializers?: Serializers;\n defaultContentType?: string;\n };\n };\n errorHandlers?: ErrorHandlers;\n showTrace?: boolean;\n } = {},\n ) => {\n const router = new NodeRouter<Request, Response, Schema>({\n onNoMatch,\n onError: onError(options.errorHandlers || [], options.showTrace || false),\n });\n\n return router\n .use(httpHeaderNormalizerMiddleware(options?.middlewares?.[\"http-header-normalizer\"] || {}))\n .use(\n serializersMiddleware(\n options?.middlewares?.serializers?.serializers || [],\n options?.middlewares?.serializers?.defaultContentType || \"application/json; charset=utf-8\",\n ),\n );\n};\n\nexport default createNodeRouter;\n","import type {\n FunctionLike, Nextable, Route, ValueOrPromise,\n} from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport JsonapiErrorHandler from \"../error-handler/jsonapi-error-handler\";\nimport ProblemErrorHandler from \"../error-handler/problem-error-handler\";\nimport type { ErrorHandler, ErrorHandlers } from \"../error-handler/types\";\n\n// eslint-disable-next-line unicorn/consistent-function-scoping,max-len\nexport const onError = <Request extends IncomingMessage, Response extends ServerResponse>(errorHandlers: ErrorHandlers, showTrace: boolean) => async (error: unknown, request: Request, response: Response): Promise<void> => {\n const apiFormat: string = request.headers.accept as string;\n\n let errorHandler: ErrorHandler = ProblemErrorHandler;\n\n if (apiFormat === \"application/vnd.api+json\") {\n errorHandler = JsonapiErrorHandler;\n }\n\n // eslint-disable-next-line no-restricted-syntax\n for (const { regex, handler } of errorHandlers) {\n if (regex.test(apiFormat)) {\n errorHandler = handler;\n break;\n }\n }\n\n // eslint-disable-next-line no-param-reassign\n (error as { expose: boolean } & Error).expose = showTrace;\n\n errorHandler(error, request, response);\n};\n\nexport const onNoMatch: <Request extends IncomingMessage, Response extends ServerResponse>(\n request: Request,\n response: Response,\n routes: Route<Nextable<FunctionLike>>[],\n) => ValueOrPromise<void> = async (request, response, routes) => {\n const uniqueMethods = [...new Set(routes.map((route) => route.method))].join(\", \");\n\n response.setHeader(\"Allow\", uniqueMethods);\n response.statusCode = 405;\n\n throw createHttpError(405, `No route with [${request.method}] method found.`);\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\nimport { ErrorSerializer, JapiError } from \"ts-japi\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultTitle = \"An error occurred\";\n\nconst jsonapiErrorHandler: ErrorHandler = (error: HttpError | JapiError | Error, _request, response) => {\n addStatusCodeToResponse(response, error);\n\n setErrorHeaders(response, error);\n\n if (error instanceof JapiError || JapiError.isLikeJapiError(error)) {\n const serializer = new ErrorSerializer();\n\n sendJson(response, serializer.serialize(error));\n } else if (error instanceof HttpError) {\n const { statusCode, title, message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: statusCode,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n } else {\n const { message } = error;\n\n sendJson(response, {\n errors: [\n {\n code: \"500\",\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n detail: message,\n },\n ],\n });\n }\n};\n\nexport default jsonapiErrorHandler;\n","import { StatusCodes } from \"http-status-codes\";\nimport type { ServerResponse } from \"node:http\";\n\nexport const setErrorHeaders = (response: ServerResponse, error: any) => {\n const headers: { [key: string]: number | string | ReadonlyArray<string> } = error.headers || {};\n\n Object.keys(headers).forEach((header: string) => {\n response.setHeader(header, headers[header] as number | string | ReadonlyArray<string>);\n });\n};\n\n/**\n * Send `JSON` object\n * @param response response object\n * @param jsonBody of data\n */\nexport const sendJson = (response: ServerResponse, jsonBody: any): void => {\n // Set header to application/json\n response.setHeader(\"content-type\", \"application/json; charset=utf-8\");\n\n response.end(JSON.stringify(jsonBody));\n};\n\nexport const addStatusCodeToResponse = (response: ServerResponse, error: any): void => {\n // respect err.statusCode\n if (error.statusCode !== undefined) {\n response.statusCode = error.statusCode;\n }\n\n // respect err.status\n if (error.status !== undefined) {\n response.statusCode = error.status;\n }\n\n // default status code to 500\n if (response.statusCode < 400) {\n response.statusCode = StatusCodes.INTERNAL_SERVER_ERROR;\n }\n};\n","import { HttpError } from \"http-errors\";\nimport { getReasonPhrase } from \"http-status-codes\";\n\nimport type { ErrorHandler } from \"./types\";\nimport { addStatusCodeToResponse, sendJson, setErrorHeaders } from \"./utils\";\n\nconst defaultType = \"https://tools.ietf.org/html/rfc2616#section-10\";\nconst defaultTitle = \"An error occurred\";\n/**\n * Normalizes errors according to the API Problem spec (RFC 7807).\n *\n * @see https://tools.ietf.org/html/rfc7807\n */\nconst problemErrorHandler: ErrorHandler = (error: HttpError | Error, _request, response) => {\n const { stack, message } = error;\n\n if (error instanceof HttpError) {\n const {\n statusCode, expose, title, type,\n } = error;\n\n response.statusCode = statusCode;\n\n setErrorHeaders(response, error);\n\n sendJson(response, {\n type: type || defaultType,\n title: title || getReasonPhrase(statusCode) || defaultTitle,\n details: message,\n ...(expose ? { trace: stack } : {}),\n });\n } else {\n addStatusCodeToResponse(response, error);\n\n sendJson(response, {\n type: defaultType,\n title: getReasonPhrase(response.statusCode) || defaultTitle,\n details: message,\n ...((error as { expose: boolean } & Error).expose ? { trace: stack } : {}),\n });\n }\n};\n\nexport default problemErrorHandler;\n","import type { NextHandler } from \"@visulima/connect\";\nimport type { IncomingHttpHeaders, IncomingMessage } from \"node:http\";\n\nconst exceptions = {\n alpn: \"ALPN\",\n \"c-pep\": \"C-PEP\",\n \"c-pep-info\": \"C-PEP-Info\",\n \"caldav-timezones\": \"CalDAV-Timezones\",\n \"content-id\": \"Content-ID\",\n \"content-md5\": \"Content-MD5\",\n dasl: \"DASL\",\n dav: \"DAV\",\n dnt: \"DNT\",\n etag: \"ETag\",\n getprofile: \"GetProfile\",\n \"http2-settings\": \"HTTP2-Settings\",\n \"last-event-id\": \"Last-Event-ID\",\n \"mime-version\": \"MIME-Version\",\n \"optional-www-authenticate\": \"Optional-WWW-Authenticate\",\n \"sec-websocket-accept\": \"Sec-WebSocket-Accept\",\n \"sec-websocket-extensions\": \"Sec-WebSocket-Extensions\",\n \"sec-webSocket-key\": \"Sec-WebSocket-Key\",\n \"sec-webSocket-protocol\": \"Sec-WebSocket-Protocol\",\n \"sec-webSocket-version\": \"Sec-WebSocket-Version\",\n slug: \"SLUG\",\n tcn: \"TCN\",\n te: \"TE\",\n ttl: \"TTL\",\n \"www-authenticate\": \"WWW-Authenticate\",\n \"x-att-deviceid\": \"X-ATT-DeviceId\",\n \"x-dnsprefetch-control\": \"X-DNSPrefetch-Control\",\n \"x-uidh\": \"X-UIDH\",\n};\n\nconst normalizeHeaderKey = (key: string, canonical: boolean) => {\n const lowerCaseKey = key.toLowerCase();\n\n if (!canonical) {\n return lowerCaseKey;\n }\n\n if (exceptions[lowerCaseKey as keyof typeof exceptions]) {\n return exceptions[lowerCaseKey as keyof typeof exceptions];\n }\n\n return (\n lowerCaseKey\n .split(\"-\")\n // eslint-disable-next-line no-unsafe-optional-chaining\n .map((text) => text[0]?.toUpperCase() + text.slice(1))\n .join(\"-\")\n );\n};\n\nconst defaults = {\n canonical: false,\n normalizeHeaderKey,\n};\n\n/**\n * HTTP headers are case-insensitive.\n * That's why NodeJS makes them lower case by default.\n * While sensible, sometimes, for example for compatibility reasons, you might need them in their more common form.\n */\nconst httpHeaderNormalizerMiddleware = (options_?: { canonical?: boolean; normalizeHeaderKey?: (key: string, canonical: boolean) => string }) => {\n const options = { ...defaults, ...options_ };\n\n return async <Request extends IncomingMessage>(request: Request, _: any, next: NextHandler) => {\n if (request.headers) {\n const rawHeaders: IncomingHttpHeaders = {};\n const headers: IncomingHttpHeaders = {};\n\n Object.keys(request.headers).forEach((key) => {\n rawHeaders[key] = request.headers[key];\n\n const normalizedKey = options.normalizeHeaderKey(key, options.canonical);\n\n if (normalizedKey !== undefined) {\n headers[normalizedKey] = request.headers[key];\n }\n });\n\n request.headers = headers;\n // @TODO at type `request.rawHeaders` to global scope\n // @ts-ignore\n request.rawHeaders = rawHeaders;\n }\n\n return next();\n };\n};\n\nexport default httpHeaderNormalizerMiddleware;\n","import type { NextHandler } from \"@visulima/connect\";\nimport accepts from \"accepts\";\nimport { header as headerCase } from \"case\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\nimport type { Serializer } from \"../serializers/types\";\nimport xmlTransformer from \"../serializers/xml\";\nimport yamlTransformer from \"../serializers/yaml\";\n\nfunction hasJsonStructure(string_: any): boolean {\n if (typeof string_ !== \"string\") {\n return false;\n }\n\n try {\n const result = JSON.parse(string_);\n const type = Object.prototype.toString.call(result);\n\n return type === \"[object Object]\" || type === \"[object Array]\";\n } catch {\n return false;\n }\n}\n\nconst contentTypeKey = \"Content-Type\";\n\n// eslint-disable-next-line max-len\nconst serialize = <Request extends IncomingMessage, Response extends ServerResponse>(\n serializers: Serializers,\n request: Request,\n response: Response | NextApiResponse,\n data: any,\n options: {\n defaultContentType: string;\n },\n // eslint-disable-next-line radar/cognitive-complexity\n) => {\n const contentType = response.getHeader(contentTypeKey) as string | undefined;\n\n // skip serialization when Content-Type is already set\n if (typeof contentType === \"string\") {\n return data;\n }\n\n const accept = accepts(request);\n const types: string[] = [...(accept.types() as string[]), options.defaultContentType];\n\n let serializedData = data;\n\n // eslint-disable-next-line no-restricted-syntax\n types.every((type) => {\n let breakTypes = false;\n\n serializers.forEach(({ regex, serializer }) => {\n if (!regex.test(type)) {\n return;\n }\n\n response.setHeader(contentTypeKey, type);\n serializedData = serializer(serializedData);\n breakTypes = true;\n });\n\n if (!breakTypes) {\n if (/yaml|yml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = yamlTransformer(hasJsonStructure(data) ? JSON.parse(data) : data);\n } else if (/xml/.test(type)) {\n response.setHeader(contentTypeKey, type);\n\n serializedData = xmlTransformer({\n [headerCase(`${request.url?.replace(\"/api/\", \"\")}`.trim())]: hasJsonStructure(data) ? JSON.parse(data) : data,\n });\n }\n }\n\n return breakTypes;\n });\n\n // eslint-disable-next-line no-param-reassign\n return serializedData;\n};\n\n// eslint-disable-next-line max-len\nconst serializersMiddleware = (serializers: Serializers = [], defaultContentType: string = \"application/json; charset=utf-8\") => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n if (typeof (response as NextApiResponse)?.send === \"function\") {\n const oldSend = (response as NextApiResponse).send;\n\n (response as NextApiResponse).send = (data) => {\n (response as NextApiResponse).send = oldSend;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n return (response as NextApiResponse).send(data);\n };\n } else {\n const oldEnd = response.end;\n\n // @ts-ignore\n response.end = (data, ...arguments_) => {\n response.end = oldEnd;\n\n // eslint-disable-next-line no-param-reassign\n data = serialize<Request, Response>(serializers, request, response, data, { defaultContentType });\n\n // @ts-ignore\n return response.end(data, ...arguments_);\n };\n }\n\n return next();\n};\n\nexport type Serializers = {\n regex: RegExp;\n serializer: Serializer;\n}[];\nexport default serializersMiddleware;\n","import type { XmlElement } from \"jstoxml\";\nimport { toXML } from \"jstoxml\";\n\nimport type { Serializer } from \"./types\";\n\nconst xmlTransformer: Serializer = (data?: XmlElement | XmlElement[]) => toXML(data, {\n header: true,\n indent: \" \",\n});\n\nexport default xmlTransformer;\n","import type { NextHandler } from \"@visulima/connect\";\nimport createHttpError from \"http-errors\";\nimport type { NextApiResponse } from \"next\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\nimport type { RateLimiterAbstract, RateLimiterRes } from \"rate-limiter-flexible\";\n\n// eslint-disable-next-line max-len\nconst getIP: (request: IncomingMessage & { ip?: string }) => string | undefined = (request) => request?.ip\n || (request.headers[\"x-forwarded-for\"] as string | undefined)\n || (request.headers[\"x-real-ip\"] as string | undefined)\n || request.connection.remoteAddress;\n\ntype HeaderValue = string | number | ReadonlyArray<string>;\n\n// eslint-disable-next-line max-len\nconst rateLimiterMiddleware = (rateLimiter: RateLimiterAbstract, headers?: (limiterResponse: RateLimiterRes) => { [key: string]: HeaderValue }) => async <Request extends IncomingMessage, Response extends ServerResponse>(request: Request, response: Response | NextApiResponse, next: NextHandler) => {\n const ip = getIP(request);\n\n if (ip === undefined) {\n throw createHttpError(400, \"Missing IP\");\n }\n\n try {\n const limiter = await rateLimiter.consume(ip);\n\n const mergedHeaders: { [key: string]: HeaderValue } = {\n \"Retry-After\": Math.round(limiter.msBeforeNext / 1000) || 1,\n \"X-RateLimit-Remaining\": limiter.remainingPoints,\n \"X-RateLimit-Reset\": new Date(Date.now() + limiter.msBeforeNext).toISOString(),\n ...headers,\n };\n\n Object.keys(mergedHeaders).forEach((key) => {\n response.setHeader(key, mergedHeaders[key] as HeaderValue);\n });\n\n await next();\n } catch {\n throw createHttpError(429, \"Too Many Requests\");\n }\n};\n\nexport default rateLimiterMiddleware;\n","import { expressWrapper } from \"@visulima/connect\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport type { CorsOptions, CorsOptionsDelegate } from \"cors\";\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport cors from \"cors\";\nimport type { IncomingMessage, ServerResponse } from \"node:http\";\n\n// eslint-disable-next-line max-len\nconst corsMiddleware = <Request extends IncomingMessage, Response extends ServerResponse>(options?: CorsOptions | CorsOptionsDelegate) => expressWrapper<Request, Response>(cors(options));\n\nexport default corsMiddleware;\n"],"mappings":"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","names":["default","HttpError","getReasonPhrase","defaultTitle","HttpError","getReasonPhrase","expressWrapper","NodeRouter","sendJson","createHttpError"]}
|
package/dist/next/cli/index.js
CHANGED
|
@@ -145,10 +145,10 @@ var groupBy = (list, keyGetter) => {
|
|
|
145
145
|
list.forEach((item) => {
|
|
146
146
|
const key = keyGetter(item);
|
|
147
147
|
const collection = map.get(key);
|
|
148
|
-
if (
|
|
149
|
-
map.set(key, [item]);
|
|
150
|
-
} else {
|
|
148
|
+
if (collection) {
|
|
151
149
|
collection.push(item);
|
|
150
|
+
} else {
|
|
151
|
+
map.set(key, [item]);
|
|
152
152
|
}
|
|
153
153
|
});
|
|
154
154
|
return map;
|
|
@@ -167,7 +167,11 @@ var listCommand = async (path = "", options = {}) => {
|
|
|
167
167
|
if (options.excludePaths) {
|
|
168
168
|
parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));
|
|
169
169
|
}
|
|
170
|
-
if (
|
|
170
|
+
if (options.group === void 0) {
|
|
171
|
+
routes_render_default([], options).forEach((renderedRoute) => {
|
|
172
|
+
console.log(renderedRoute);
|
|
173
|
+
});
|
|
174
|
+
} else {
|
|
171
175
|
const groupedMap = groupBy(parsedApiRoutes, (route) => {
|
|
172
176
|
if (options.group === "path") {
|
|
173
177
|
return route.path.replace("/pages", "").split("/")[1];
|
|
@@ -187,10 +191,6 @@ var listCommand = async (path = "", options = {}) => {
|
|
|
187
191
|
});
|
|
188
192
|
counter += 1;
|
|
189
193
|
});
|
|
190
|
-
} else {
|
|
191
|
-
routes_render_default([], options).forEach((renderedRoute) => {
|
|
192
|
-
console.log(renderedRoute);
|
|
193
|
-
});
|
|
194
194
|
}
|
|
195
195
|
console.log(`
|
|
196
196
|
Listed ${_chalk2.default.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? "route" : "routes"}.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../src/next/cli/list/list-command.ts","../../../src/next/cli/list/api-route-file-parser.ts","../../../src/next/cli/list/collect-api-route-files.ts","../../../src/next/cli/list/routes-render.ts"],"names":["colors","join","process","fs"],"mappings":"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","sourcesContent":["import colors from \"chalk\";\nimport { join } from \"node:path\";\nimport process from \"node:process\";\n\nimport apiRouteFileParser from \"./api-route-file-parser\";\nimport collectApiRouteFiles from \"./collect-api-route-files\";\nimport routesRender from \"./routes-render\";\nimport type { Route } from \"./types\";\n\ntype RenderOptions = {\n includePaths: string[];\n excludePaths: string[];\n methods: string[];\n group: string;\n};\n\nconst groupBy = (list: Route[], keyGetter: (item: Route) => keyof Route): Map<string, Route[]> => {\n const map = new Map();\n\n list.forEach((item) => {\n const key = keyGetter(item);\n const collection = map.get(key);\n\n if (!collection) {\n map.set(key, [item]);\n } else {\n collection.push(item);\n }\n });\n\n return map;\n};\n\nconst listCommand = async (\n path: string = \"\",\n options: Partial<\n {\n verbose: boolean;\n } & RenderOptions\n > = {},\n// eslint-disable-next-line radar/cognitive-complexity\n) => {\n const routePath = join(process.cwd(), path);\n\n const apiRouteFiles = await collectApiRouteFiles(routePath, options.verbose || false);\n\n if (apiRouteFiles.length === 0) {\n // eslint-disable-next-line no-console\n console.error(`No API routes found, in \"${routePath}\".`);\n // eslint-disable-next-line unicorn/no-process-exit\n process.exit(1);\n }\n\n let parsedApiRoutes = apiRouteFiles.flatMap((apiRouteFile) => apiRouteFileParser(apiRouteFile, routePath, options.verbose || false));\n\n if (options.includePaths) {\n parsedApiRoutes = options.includePaths.flatMap((ipath) => parsedApiRoutes.filter((route) => route.path.startsWith(ipath)));\n }\n\n if (options.excludePaths) {\n parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));\n }\n\n if (typeof options.group !== \"undefined\") {\n const groupedMap = groupBy(parsedApiRoutes, (route) => {\n if (options.group === \"path\") {\n return route.path.replace(\"/pages\", \"\").split(\"/\")[1];\n }\n\n return route.tags[0] || \"unsorted\";\n });\n\n let counter = 0;\n\n groupedMap.forEach((routes, key) => {\n if (counter > 0) {\n // eslint-disable-next-line no-console\n console.log();\n }\n\n const dotsCount = (process.stdout.columns - 16 - key.length) / 2;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\" \").join(\"\") : \"\";\n // eslint-disable-next-line no-console\n console.log(dots + colors.bold.underline(key));\n\n routesRender(routes, options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n\n counter += 1;\n });\n } else {\n routesRender([], options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n }\n // });\n\n // eslint-disable-next-line no-console\n console.log(`\\n Listed ${colors.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? \"route\" : \"routes\"}.\\n`);\n};\n\nexport default listCommand;\n","import type { OpenApiObject } from \"@visulima/jsdoc-open-api\";\nimport { jsDocumentCommentsToOpenApi, parseFile, swaggerJsDocumentCommentsToOpenApi } from \"@visulima/jsdoc-open-api\";\nimport fs from \"node:fs\";\nimport process from \"node:process\";\n\nimport type { Route } from \"./types\";\n\nconst extensionRegex = /\\.(js|ts|mjs|cjs)$/;\n\nconst apiRouteFileParser = (apiRouteFile: string, cwdPath: string, verbose: boolean = false): Route[] => {\n let specs: OpenApiObject[] = [];\n\n const parsedJsDocumentFile = parseFile(apiRouteFile, jsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedJsDocumentFile.map((item) => item.spec)];\n\n const parsedSwaggerJsDocumentFile = parseFile(apiRouteFile, swaggerJsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedSwaggerJsDocumentFile.map((item) => item.spec)];\n\n const routes: Route[] = [];\n\n if (specs.length === 0) {\n const apiRouteFileContent = fs.readFileSync(apiRouteFile, \"utf8\");\n\n apiRouteFileContent.split(/\\r?\\n/).forEach((line) => {\n const match = line.match(/[=aces|]+\\s[\"'|](GET|POST|PUT|PATCH|HEAD|DELETE|OPTIONS)[\"'|]/);\n\n if (match) {\n let [, method] = match;\n\n if (method === \"GET\") {\n method = \"GET|HEAD\";\n }\n\n routes.push({\n method: method as string,\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n });\n\n if (routes.length === 0) {\n routes.push({\n method: \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\",\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n\n return routes;\n }\n\n specs.forEach((spec) => {\n const paths = Object.entries(spec.paths);\n\n paths.forEach(([path, pathSpec]) => {\n const methods = Object.entries(pathSpec);\n\n methods.forEach(([method, methodSpec]) => {\n routes.push({\n path, method: method.toUpperCase(), tags: methodSpec.tags, file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n });\n });\n });\n\n return routes;\n};\n\nexport default apiRouteFileParser;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport { collect } from \"@visulima/readdir\";\nimport fs from \"node:fs\";\nimport { join } from \"node:path\";\n\nconst isDirectory = (path: string): boolean => {\n try {\n return fs.statSync(path).isDirectory();\n } catch {\n return false;\n }\n};\n\nconst collectApiRouteFiles = async (path: string = \"\", verbose: boolean = false): Promise<string[]> => {\n let apiFolderPath = join(path, \"pages/api\");\n\n // src/pages will be ignored if pages is present in the root directory\n if (!isDirectory(apiFolderPath)) {\n apiFolderPath = join(path, \"src/pages/api\");\n\n if (!isDirectory(apiFolderPath)) {\n return [];\n }\n }\n\n return collect(apiFolderPath, {\n extensions: [\".js\", \".cjs\", \".mjs\", \".ts\"],\n includeDirs: false,\n minimatchOptions: {\n match: {\n debug: verbose,\n matchBase: true,\n },\n skip: {\n debug: verbose,\n matchBase: true,\n },\n },\n });\n};\n\nexport default collectApiRouteFiles;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\n\nimport type { Route } from \"./types\";\n\nconst renderRoute = (method: string, routePath: string): string => {\n const colorMap = {\n GET: colors.blue,\n POST: colors.yellow,\n PATCH: colors.yellow,\n PUT: colors.yellow,\n DELETE: colors.redBright,\n OPTIONS: colors.hex(\"#6C7280\"),\n ANY: colors.redBright,\n HEAD: colors.hex(\"#6C7280\"),\n };\n\n let methodText: string;\n\n if (method === \"GET|HEAD\") {\n methodText = `${colors.blue(\"GET\")}${colors.grey(\"|HEAD\")}`;\n } else {\n const coloredMethod = colorMap[method as keyof typeof colorMap](method);\n\n methodText = method === \"GET\" ? `${coloredMethod}${colors.grey(\"|HEAD\")}` : coloredMethod;\n }\n\n const spacesCount = method === \"GET\" ? 6 : 14 - method.length;\n const spaces = Array.from({ length: spacesCount }).fill(\" \").join(\"\");\n\n const dotsCount = process.stdout.columns - 16 - routePath.length - 4;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\".\").join(\"\") : \"\";\n\n const routeText = routePath\n .split(\"/\")\n .map((segment) => {\n const isDynamicSegment = [\"[\", \":\"].includes(segment[0] || \"\");\n\n return isDynamicSegment ? colors.yellowBright(segment) : segment;\n })\n .join(\"/\");\n\n return ` ${methodText}${spaces}${routeText}${colors.grey(dots)}`;\n};\n\nconst routesRender = (routesMap: Route[], options: { methods?: string[]; } = {}) => routesMap\n .map((route) => {\n if (options.methods && options.methods.includes(route.method)) {\n return;\n }\n\n if (route.method === \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\") {\n // eslint-disable-next-line no-param-reassign\n route.method = \"ANY\";\n }\n\n // eslint-disable-next-line consistent-return\n return renderRoute(route.method, route.path.replace(\"/pages\", \"\"));\n })\n .filter(Boolean);\n\nexport default routesRender;\n"]}
|
|
1
|
+
{"version":3,"sources":["../../../src/next/cli/list/list-command.ts","../../../src/next/cli/list/api-route-file-parser.ts","../../../src/next/cli/list/collect-api-route-files.ts","../../../src/next/cli/list/routes-render.ts"],"names":["colors","join","process","fs"],"mappings":"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","sourcesContent":["// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\nimport { join } from \"node:path\";\nimport process from \"node:process\";\n\nimport apiRouteFileParser from \"./api-route-file-parser\";\nimport collectApiRouteFiles from \"./collect-api-route-files\";\nimport routesRender from \"./routes-render\";\nimport type { Route } from \"./types\";\n\ntype RenderOptions = {\n includePaths: string[];\n excludePaths: string[];\n methods: string[];\n group: string;\n};\n\nconst groupBy = (list: Route[], keyGetter: (item: Route) => keyof Route): Map<string, Route[]> => {\n const map = new Map();\n\n list.forEach((item) => {\n const key = keyGetter(item);\n const collection = map.get(key);\n\n if (collection) {\n collection.push(item);\n } else {\n map.set(key, [item]);\n }\n });\n\n return map;\n};\n\nconst listCommand = async (\n path: string = \"\",\n options: Partial<\n {\n verbose: boolean;\n } & RenderOptions\n > = {},\n// eslint-disable-next-line radar/cognitive-complexity\n) => {\n const routePath = join(process.cwd(), path);\n\n const apiRouteFiles = await collectApiRouteFiles(routePath, options.verbose || false);\n\n if (apiRouteFiles.length === 0) {\n // eslint-disable-next-line no-console\n console.error(`No API routes found, in \"${routePath}\".`);\n // eslint-disable-next-line unicorn/no-process-exit\n process.exit(1);\n }\n\n let parsedApiRoutes = apiRouteFiles.flatMap((apiRouteFile) => apiRouteFileParser(apiRouteFile, routePath, options.verbose || false));\n\n if (options.includePaths) {\n parsedApiRoutes = options.includePaths.flatMap((ipath) => parsedApiRoutes.filter((route) => route.path.startsWith(ipath)));\n }\n\n if (options.excludePaths) {\n parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));\n }\n\n if (options.group === undefined) {\n routesRender([], options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n } else {\n const groupedMap = groupBy(parsedApiRoutes, (route) => {\n if (options.group === \"path\") {\n return route.path.replace(\"/pages\", \"\").split(\"/\")[1];\n }\n\n return route.tags[0] || \"unsorted\";\n });\n\n let counter = 0;\n\n groupedMap.forEach((routes, key) => {\n if (counter > 0) {\n // eslint-disable-next-line no-console\n console.log();\n }\n\n const dotsCount = (process.stdout.columns - 16 - key.length) / 2;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\" \").join(\"\") : \"\";\n // eslint-disable-next-line no-console\n console.log(dots + colors.bold.underline(key));\n\n routesRender(routes, options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n\n counter += 1;\n });\n }\n // });\n\n // eslint-disable-next-line no-console\n console.log(`\\n Listed ${colors.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? \"route\" : \"routes\"}.\\n`);\n};\n\nexport default listCommand;\n","import type { OpenApiObject } from \"@visulima/jsdoc-open-api\";\nimport { jsDocumentCommentsToOpenApi, parseFile, swaggerJsDocumentCommentsToOpenApi } from \"@visulima/jsdoc-open-api\";\nimport fs from \"node:fs\";\nimport process from \"node:process\";\n\nimport type { Route } from \"./types\";\n\nconst extensionRegex = /\\.(js|ts|mjs|cjs)$/;\n\nconst apiRouteFileParser = (apiRouteFile: string, cwdPath: string, verbose: boolean = false): Route[] => {\n let specs: OpenApiObject[] = [];\n\n const parsedJsDocumentFile = parseFile(apiRouteFile, jsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedJsDocumentFile.map((item) => item.spec)];\n\n const parsedSwaggerJsDocumentFile = parseFile(apiRouteFile, swaggerJsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedSwaggerJsDocumentFile.map((item) => item.spec)];\n\n const routes: Route[] = [];\n\n if (specs.length === 0) {\n const apiRouteFileContent = fs.readFileSync(apiRouteFile, \"utf8\");\n\n apiRouteFileContent.split(/\\r?\\n/).forEach((line) => {\n const match = line.match(/[=aces|]+\\s[\"'|](GET|POST|PUT|PATCH|HEAD|DELETE|OPTIONS)[\"'|]/);\n\n if (match) {\n let [, method] = match;\n\n if (method === \"GET\") {\n method = \"GET|HEAD\";\n }\n\n routes.push({\n method: method as string,\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n });\n\n if (routes.length === 0) {\n routes.push({\n method: \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\",\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n\n return routes;\n }\n\n specs.forEach((spec) => {\n const paths = Object.entries(spec.paths);\n\n paths.forEach(([path, pathSpec]) => {\n const methods = Object.entries(pathSpec);\n\n methods.forEach(([method, methodSpec]) => {\n routes.push({\n path, method: method.toUpperCase(), tags: methodSpec.tags, file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n });\n });\n });\n\n return routes;\n};\n\nexport default apiRouteFileParser;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport { collect } from \"@visulima/readdir\";\nimport fs from \"node:fs\";\nimport { join } from \"node:path\";\n\nconst isDirectory = (path: string): boolean => {\n try {\n return fs.statSync(path).isDirectory();\n } catch {\n return false;\n }\n};\n\nconst collectApiRouteFiles = async (path: string = \"\", verbose: boolean = false): Promise<string[]> => {\n let apiFolderPath = join(path, \"pages/api\");\n\n // src/pages will be ignored if pages is present in the root directory\n if (!isDirectory(apiFolderPath)) {\n apiFolderPath = join(path, \"src/pages/api\");\n\n if (!isDirectory(apiFolderPath)) {\n return [];\n }\n }\n\n return collect(apiFolderPath, {\n extensions: [\".js\", \".cjs\", \".mjs\", \".ts\"],\n includeDirs: false,\n minimatchOptions: {\n match: {\n debug: verbose,\n matchBase: true,\n },\n skip: {\n debug: verbose,\n matchBase: true,\n },\n },\n });\n};\n\nexport default collectApiRouteFiles;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\n\nimport type { Route } from \"./types\";\n\nconst renderRoute = (method: string, routePath: string): string => {\n const colorMap = {\n GET: colors.blue,\n POST: colors.yellow,\n PATCH: colors.yellow,\n PUT: colors.yellow,\n DELETE: colors.redBright,\n OPTIONS: colors.hex(\"#6C7280\"),\n ANY: colors.redBright,\n HEAD: colors.hex(\"#6C7280\"),\n };\n\n let methodText: string;\n\n if (method === \"GET|HEAD\") {\n methodText = `${colors.blue(\"GET\")}${colors.grey(\"|HEAD\")}`;\n } else {\n const coloredMethod = colorMap[method as keyof typeof colorMap](method);\n\n methodText = method === \"GET\" ? `${coloredMethod}${colors.grey(\"|HEAD\")}` : coloredMethod;\n }\n\n const spacesCount = method === \"GET\" ? 6 : 14 - method.length;\n const spaces = Array.from({ length: spacesCount }).fill(\" \").join(\"\");\n\n const dotsCount = process.stdout.columns - 16 - routePath.length - 4;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\".\").join(\"\") : \"\";\n\n const routeText = routePath\n .split(\"/\")\n .map((segment) => {\n const isDynamicSegment = [\"[\", \":\"].includes(segment[0] || \"\");\n\n return isDynamicSegment ? colors.yellowBright(segment) : segment;\n })\n .join(\"/\");\n\n return ` ${methodText}${spaces}${routeText}${colors.grey(dots)}`;\n};\n\nconst routesRender = (routesMap: Route[], options: { methods?: string[]; } = {}) => routesMap\n .map((route) => {\n if (options.methods && options.methods.includes(route.method)) {\n return;\n }\n\n if (route.method === \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\") {\n // eslint-disable-next-line no-param-reassign\n route.method = \"ANY\";\n }\n\n // eslint-disable-next-line consistent-return\n return renderRoute(route.method, route.path.replace(\"/pages\", \"\"));\n })\n .filter(Boolean);\n\nexport default routesRender;\n"]}
|
package/dist/next/cli/index.mjs
CHANGED
|
@@ -145,10 +145,10 @@ var groupBy = (list, keyGetter) => {
|
|
|
145
145
|
list.forEach((item) => {
|
|
146
146
|
const key = keyGetter(item);
|
|
147
147
|
const collection = map.get(key);
|
|
148
|
-
if (
|
|
149
|
-
map.set(key, [item]);
|
|
150
|
-
} else {
|
|
148
|
+
if (collection) {
|
|
151
149
|
collection.push(item);
|
|
150
|
+
} else {
|
|
151
|
+
map.set(key, [item]);
|
|
152
152
|
}
|
|
153
153
|
});
|
|
154
154
|
return map;
|
|
@@ -167,7 +167,11 @@ var listCommand = async (path = "", options = {}) => {
|
|
|
167
167
|
if (options.excludePaths) {
|
|
168
168
|
parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));
|
|
169
169
|
}
|
|
170
|
-
if (
|
|
170
|
+
if (options.group === void 0) {
|
|
171
|
+
routes_render_default([], options).forEach((renderedRoute) => {
|
|
172
|
+
console.log(renderedRoute);
|
|
173
|
+
});
|
|
174
|
+
} else {
|
|
171
175
|
const groupedMap = groupBy(parsedApiRoutes, (route) => {
|
|
172
176
|
if (options.group === "path") {
|
|
173
177
|
return route.path.replace("/pages", "").split("/")[1];
|
|
@@ -187,10 +191,6 @@ var listCommand = async (path = "", options = {}) => {
|
|
|
187
191
|
});
|
|
188
192
|
counter += 1;
|
|
189
193
|
});
|
|
190
|
-
} else {
|
|
191
|
-
routes_render_default([], options).forEach((renderedRoute) => {
|
|
192
|
-
console.log(renderedRoute);
|
|
193
|
-
});
|
|
194
194
|
}
|
|
195
195
|
console.log(`
|
|
196
196
|
Listed ${colors2.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? "route" : "routes"}.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../../src/next/cli/list/list-command.ts","../../../src/next/cli/list/api-route-file-parser.ts","../../../src/next/cli/list/collect-api-route-files.ts","../../../src/next/cli/list/routes-render.ts"],"sourcesContent":["import colors from \"chalk\";\nimport { join } from \"node:path\";\nimport process from \"node:process\";\n\nimport apiRouteFileParser from \"./api-route-file-parser\";\nimport collectApiRouteFiles from \"./collect-api-route-files\";\nimport routesRender from \"./routes-render\";\nimport type { Route } from \"./types\";\n\ntype RenderOptions = {\n includePaths: string[];\n excludePaths: string[];\n methods: string[];\n group: string;\n};\n\nconst groupBy = (list: Route[], keyGetter: (item: Route) => keyof Route): Map<string, Route[]> => {\n const map = new Map();\n\n list.forEach((item) => {\n const key = keyGetter(item);\n const collection = map.get(key);\n\n if (!collection) {\n map.set(key, [item]);\n } else {\n collection.push(item);\n }\n });\n\n return map;\n};\n\nconst listCommand = async (\n path: string = \"\",\n options: Partial<\n {\n verbose: boolean;\n } & RenderOptions\n > = {},\n// eslint-disable-next-line radar/cognitive-complexity\n) => {\n const routePath = join(process.cwd(), path);\n\n const apiRouteFiles = await collectApiRouteFiles(routePath, options.verbose || false);\n\n if (apiRouteFiles.length === 0) {\n // eslint-disable-next-line no-console\n console.error(`No API routes found, in \"${routePath}\".`);\n // eslint-disable-next-line unicorn/no-process-exit\n process.exit(1);\n }\n\n let parsedApiRoutes = apiRouteFiles.flatMap((apiRouteFile) => apiRouteFileParser(apiRouteFile, routePath, options.verbose || false));\n\n if (options.includePaths) {\n parsedApiRoutes = options.includePaths.flatMap((ipath) => parsedApiRoutes.filter((route) => route.path.startsWith(ipath)));\n }\n\n if (options.excludePaths) {\n parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));\n }\n\n if (typeof options.group !== \"undefined\") {\n const groupedMap = groupBy(parsedApiRoutes, (route) => {\n if (options.group === \"path\") {\n return route.path.replace(\"/pages\", \"\").split(\"/\")[1];\n }\n\n return route.tags[0] || \"unsorted\";\n });\n\n let counter = 0;\n\n groupedMap.forEach((routes, key) => {\n if (counter > 0) {\n // eslint-disable-next-line no-console\n console.log();\n }\n\n const dotsCount = (process.stdout.columns - 16 - key.length) / 2;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\" \").join(\"\") : \"\";\n // eslint-disable-next-line no-console\n console.log(dots + colors.bold.underline(key));\n\n routesRender(routes, options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n\n counter += 1;\n });\n } else {\n routesRender([], options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n }\n // });\n\n // eslint-disable-next-line no-console\n console.log(`\\n Listed ${colors.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? \"route\" : \"routes\"}.\\n`);\n};\n\nexport default listCommand;\n","import type { OpenApiObject } from \"@visulima/jsdoc-open-api\";\nimport { jsDocumentCommentsToOpenApi, parseFile, swaggerJsDocumentCommentsToOpenApi } from \"@visulima/jsdoc-open-api\";\nimport fs from \"node:fs\";\nimport process from \"node:process\";\n\nimport type { Route } from \"./types\";\n\nconst extensionRegex = /\\.(js|ts|mjs|cjs)$/;\n\nconst apiRouteFileParser = (apiRouteFile: string, cwdPath: string, verbose: boolean = false): Route[] => {\n let specs: OpenApiObject[] = [];\n\n const parsedJsDocumentFile = parseFile(apiRouteFile, jsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedJsDocumentFile.map((item) => item.spec)];\n\n const parsedSwaggerJsDocumentFile = parseFile(apiRouteFile, swaggerJsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedSwaggerJsDocumentFile.map((item) => item.spec)];\n\n const routes: Route[] = [];\n\n if (specs.length === 0) {\n const apiRouteFileContent = fs.readFileSync(apiRouteFile, \"utf8\");\n\n apiRouteFileContent.split(/\\r?\\n/).forEach((line) => {\n const match = line.match(/[=aces|]+\\s[\"'|](GET|POST|PUT|PATCH|HEAD|DELETE|OPTIONS)[\"'|]/);\n\n if (match) {\n let [, method] = match;\n\n if (method === \"GET\") {\n method = \"GET|HEAD\";\n }\n\n routes.push({\n method: method as string,\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n });\n\n if (routes.length === 0) {\n routes.push({\n method: \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\",\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n\n return routes;\n }\n\n specs.forEach((spec) => {\n const paths = Object.entries(spec.paths);\n\n paths.forEach(([path, pathSpec]) => {\n const methods = Object.entries(pathSpec);\n\n methods.forEach(([method, methodSpec]) => {\n routes.push({\n path, method: method.toUpperCase(), tags: methodSpec.tags, file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n });\n });\n });\n\n return routes;\n};\n\nexport default apiRouteFileParser;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport { collect } from \"@visulima/readdir\";\nimport fs from \"node:fs\";\nimport { join } from \"node:path\";\n\nconst isDirectory = (path: string): boolean => {\n try {\n return fs.statSync(path).isDirectory();\n } catch {\n return false;\n }\n};\n\nconst collectApiRouteFiles = async (path: string = \"\", verbose: boolean = false): Promise<string[]> => {\n let apiFolderPath = join(path, \"pages/api\");\n\n // src/pages will be ignored if pages is present in the root directory\n if (!isDirectory(apiFolderPath)) {\n apiFolderPath = join(path, \"src/pages/api\");\n\n if (!isDirectory(apiFolderPath)) {\n return [];\n }\n }\n\n return collect(apiFolderPath, {\n extensions: [\".js\", \".cjs\", \".mjs\", \".ts\"],\n includeDirs: false,\n minimatchOptions: {\n match: {\n debug: verbose,\n matchBase: true,\n },\n skip: {\n debug: verbose,\n matchBase: true,\n },\n },\n });\n};\n\nexport default collectApiRouteFiles;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\n\nimport type { Route } from \"./types\";\n\nconst renderRoute = (method: string, routePath: string): string => {\n const colorMap = {\n GET: colors.blue,\n POST: colors.yellow,\n PATCH: colors.yellow,\n PUT: colors.yellow,\n DELETE: colors.redBright,\n OPTIONS: colors.hex(\"#6C7280\"),\n ANY: colors.redBright,\n HEAD: colors.hex(\"#6C7280\"),\n };\n\n let methodText: string;\n\n if (method === \"GET|HEAD\") {\n methodText = `${colors.blue(\"GET\")}${colors.grey(\"|HEAD\")}`;\n } else {\n const coloredMethod = colorMap[method as keyof typeof colorMap](method);\n\n methodText = method === \"GET\" ? `${coloredMethod}${colors.grey(\"|HEAD\")}` : coloredMethod;\n }\n\n const spacesCount = method === \"GET\" ? 6 : 14 - method.length;\n const spaces = Array.from({ length: spacesCount }).fill(\" \").join(\"\");\n\n const dotsCount = process.stdout.columns - 16 - routePath.length - 4;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\".\").join(\"\") : \"\";\n\n const routeText = routePath\n .split(\"/\")\n .map((segment) => {\n const isDynamicSegment = [\"[\", \":\"].includes(segment[0] || \"\");\n\n return isDynamicSegment ? colors.yellowBright(segment) : segment;\n })\n .join(\"/\");\n\n return ` ${methodText}${spaces}${routeText}${colors.grey(dots)}`;\n};\n\nconst routesRender = (routesMap: Route[], options: { methods?: string[]; } = {}) => routesMap\n .map((route) => {\n if (options.methods && options.methods.includes(route.method)) {\n return;\n }\n\n if (route.method === \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\") {\n // eslint-disable-next-line no-param-reassign\n route.method = \"ANY\";\n }\n\n // eslint-disable-next-line consistent-return\n return renderRoute(route.method, route.path.replace(\"/pages\", \"\"));\n })\n .filter(Boolean);\n\nexport default routesRender;\n"],"mappings":";;;AAAA,OAAOA,aAAY;AACnB,SAAS,QAAAC,aAAY;AACrB,OAAOC,cAAa;;;ACDpB,SAAS,6BAA6B,WAAW,0CAA0C;AAC3F,OAAO,QAAQ;AACf,OAAOC,cAAa;AAIpB,IAAM,iBAAiB;AAEvB,IAAM,qBAAqB,CAAC,cAAsB,SAAiB,UAAmB,UAAmB;AACrG,MAAI,QAAyB,CAAC;AAE9B,QAAM,uBAAuB,UAAU,cAAc,6BAA6B,OAAO;AAEzF,UAAQ,CAAC,GAAG,OAAO,GAAG,qBAAqB,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC;AAEnE,QAAM,8BAA8B,UAAU,cAAc,oCAAoC,OAAO;AAEvG,UAAQ,CAAC,GAAG,OAAO,GAAG,4BAA4B,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC;AAE1E,QAAM,SAAkB,CAAC;AAEzB,MAAI,MAAM,WAAW,GAAG;AACpB,UAAM,sBAAsB,GAAG,aAAa,cAAc,MAAM;AAEhE,wBAAoB,MAAM,OAAO,EAAE,QAAQ,CAAC,SAAS;AACjD,YAAM,QAAQ,KAAK,MAAM,+DAA+D;AAExF,UAAI,OAAO;AACP,YAAI,CAAC,EAAE,MAAM,IAAI;AAEjB,YAAI,WAAW,OAAO;AAClB,mBAAS;AAAA,QACb;AAEA,eAAO,KAAK;AAAA,UACR;AAAA,UACA,MAAM,aAAa,QAAQ,SAAS,EAAE,EAAE,QAAQ,gBAAgB,EAAE;AAAA,UAClE,MAAM,CAAC;AAAA,UACP,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,QACtD,CAAC;AAAA,MACL;AAAA,IACJ,CAAC;AAED,QAAI,OAAO,WAAW,GAAG;AACrB,aAAO,KAAK;AAAA,QACR,QAAQ;AAAA,QACR,MAAM,aAAa,QAAQ,SAAS,EAAE,EAAE,QAAQ,gBAAgB,EAAE;AAAA,QAClE,MAAM,CAAC;AAAA,QACP,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,MACtD,CAAC;AAAA,IACL;AAEA,WAAO;AAAA,EACX;AAEA,QAAM,QAAQ,CAAC,SAAS;AACpB,UAAM,QAAQ,OAAO,QAAQ,KAAK,KAAK;AAEvC,UAAM,QAAQ,CAAC,CAAC,MAAM,QAAQ,MAAM;AAChC,YAAM,UAAU,OAAO,QAAQ,QAAQ;AAEvC,cAAQ,QAAQ,CAAC,CAAC,QAAQ,UAAU,MAAM;AACtC,eAAO,KAAK;AAAA,UACR;AAAA,UAAM,QAAQ,OAAO,YAAY;AAAA,UAAG,MAAM,WAAW;AAAA,UAAM,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,QACjH,CAAC;AAAA,MACL,CAAC;AAAA,IACL,CAAC;AAAA,EACL,CAAC;AAED,SAAO;AACX;AAEA,IAAO,gCAAQ;;;ACxEf,SAAS,eAAe;AACxB,OAAOC,SAAQ;AACf,SAAS,YAAY;AAErB,IAAM,cAAc,CAAC,SAA0B;AAC3C,MAAI;AACA,WAAOA,IAAG,SAAS,IAAI,EAAE,YAAY;AAAA,EACzC,QAAE;AACE,WAAO;AAAA,EACX;AACJ;AAEA,IAAM,uBAAuB,OAAO,OAAe,IAAI,UAAmB,UAA6B;AACnG,MAAI,gBAAgB,KAAK,MAAM,WAAW;AAG1C,MAAI,CAAC,YAAY,aAAa,GAAG;AAC7B,oBAAgB,KAAK,MAAM,eAAe;AAE1C,QAAI,CAAC,YAAY,aAAa,GAAG;AAC7B,aAAO,CAAC;AAAA,IACZ;AAAA,EACJ;AAEA,SAAO,QAAQ,eAAe;AAAA,IAC1B,YAAY,CAAC,OAAO,QAAQ,QAAQ,KAAK;AAAA,IACzC,aAAa;AAAA,IACb,kBAAkB;AAAA,MACd,OAAO;AAAA,QACH,OAAO;AAAA,QACP,WAAW;AAAA,MACf;AAAA,MACA,MAAM;AAAA,QACF,OAAO;AAAA,QACP,WAAW;AAAA,MACf;AAAA,IACJ;AAAA,EACJ,CAAC;AACL;AAEA,IAAO,kCAAQ;;;ACxCf,OAAO,YAAY;AAInB,IAAM,cAAc,CAAC,QAAgB,cAA8B;AAC/D,QAAM,WAAW;AAAA,IACb,KAAK,OAAO;AAAA,IACZ,MAAM,OAAO;AAAA,IACb,OAAO,OAAO;AAAA,IACd,KAAK,OAAO;AAAA,IACZ,QAAQ,OAAO;AAAA,IACf,SAAS,OAAO,IAAI,SAAS;AAAA,IAC7B,KAAK,OAAO;AAAA,IACZ,MAAM,OAAO,IAAI,SAAS;AAAA,EAC9B;AAEA,MAAI;AAEJ,MAAI,WAAW,YAAY;AACvB,iBAAa,GAAG,OAAO,KAAK,KAAK,IAAI,OAAO,KAAK,OAAO;AAAA,EAC5D,OAAO;AACH,UAAM,gBAAgB,SAAS,QAAiC,MAAM;AAEtE,iBAAa,WAAW,QAAQ,GAAG,gBAAgB,OAAO,KAAK,OAAO,MAAM;AAAA,EAChF;AAEA,QAAM,cAAc,WAAW,QAAQ,IAAI,KAAK,OAAO;AACvD,QAAM,SAAS,MAAM,KAAK,EAAE,QAAQ,YAAY,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE;AAEpE,QAAM,YAAY,QAAQ,OAAO,UAAU,KAAK,UAAU,SAAS;AACnE,QAAM,OAAO,YAAY,IAAI,MAAM,KAAK,EAAE,QAAQ,UAAU,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE,IAAI;AAEpF,QAAM,YAAY,UACb,MAAM,GAAG,EACT,IAAI,CAAC,YAAY;AACd,UAAM,mBAAmB,CAAC,KAAK,GAAG,EAAE,SAAS,QAAQ,MAAM,EAAE;AAE7D,WAAO,mBAAmB,OAAO,aAAa,OAAO,IAAI;AAAA,EAC7D,CAAC,EACA,KAAK,GAAG;AAEb,SAAO,KAAK,aAAa,SAAS,YAAY,OAAO,KAAK,IAAI;AAClE;AAEA,IAAM,eAAe,CAAC,WAAoB,UAAmC,CAAC,MAAM,UAC/E,IAAI,CAAC,UAAU;AACZ,MAAI,QAAQ,WAAW,QAAQ,QAAQ,SAAS,MAAM,MAAM,GAAG;AAC3D;AAAA,EACJ;AAEA,MAAI,MAAM,WAAW,0CAA0C;AAE3D,UAAM,SAAS;AAAA,EACnB;AAGA,SAAO,YAAY,MAAM,QAAQ,MAAM,KAAK,QAAQ,UAAU,EAAE,CAAC;AACrE,CAAC,EACA,OAAO,OAAO;AAEnB,IAAO,wBAAQ;;;AH7Cf,IAAM,UAAU,CAAC,MAAe,cAAkE;AAC9F,QAAM,MAAM,oBAAI,IAAI;AAEpB,OAAK,QAAQ,CAAC,SAAS;AACnB,UAAM,MAAM,UAAU,IAAI;AAC1B,UAAM,aAAa,IAAI,IAAI,GAAG;AAE9B,QAAI,CAAC,YAAY;AACb,UAAI,IAAI,KAAK,CAAC,IAAI,CAAC;AAAA,IACvB,OAAO;AACH,iBAAW,KAAK,IAAI;AAAA,IACxB;AAAA,EACJ,CAAC;AAED,SAAO;AACX;AAEA,IAAM,cAAc,OAChB,OAAe,IACf,UAII,CAAC,MAEJ;AACD,QAAM,YAAYC,MAAKC,SAAQ,IAAI,GAAG,IAAI;AAE1C,QAAM,gBAAgB,MAAM,gCAAqB,WAAW,QAAQ,WAAW,KAAK;AAEpF,MAAI,cAAc,WAAW,GAAG;AAE5B,YAAQ,MAAM,4BAA4B,aAAa;AAEvD,IAAAA,SAAQ,KAAK,CAAC;AAAA,EAClB;AAEA,MAAI,kBAAkB,cAAc,QAAQ,CAAC,iBAAiB,8BAAmB,cAAc,WAAW,QAAQ,WAAW,KAAK,CAAC;AAEnI,MAAI,QAAQ,cAAc;AACtB,sBAAkB,QAAQ,aAAa,QAAQ,CAAC,UAAU,gBAAgB,OAAO,CAAC,UAAU,MAAM,KAAK,WAAW,KAAK,CAAC,CAAC;AAAA,EAC7H;AAEA,MAAI,QAAQ,cAAc;AACtB,sBAAkB,QAAQ,aAAa,QAAQ,CAAC,UAAU,gBAAgB,OAAO,CAAC,UAAU,CAAC,MAAM,KAAK,WAAW,KAAK,CAAC,CAAC;AAAA,EAC9H;AAEA,MAAI,OAAO,QAAQ,UAAU,aAAa;AACtC,UAAM,aAAa,QAAQ,iBAAiB,CAAC,UAAU;AACnD,UAAI,QAAQ,UAAU,QAAQ;AAC1B,eAAO,MAAM,KAAK,QAAQ,UAAU,EAAE,EAAE,MAAM,GAAG,EAAE;AAAA,MACvD;AAEA,aAAO,MAAM,KAAK,MAAM;AAAA,IAC5B,CAAC;AAED,QAAI,UAAU;AAEd,eAAW,QAAQ,CAAC,QAAQ,QAAQ;AAChC,UAAI,UAAU,GAAG;AAEb,gBAAQ,IAAI;AAAA,MAChB;AAEA,YAAM,aAAaA,SAAQ,OAAO,UAAU,KAAK,IAAI,UAAU;AAC/D,YAAM,OAAO,YAAY,IAAI,MAAM,KAAK,EAAE,QAAQ,UAAU,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE,IAAI;AAEpF,cAAQ,IAAI,OAAOC,QAAO,KAAK,UAAU,GAAG,CAAC;AAE7C,4BAAa,QAAQ,OAAO,EAAE,QAAQ,CAAC,kBAAkB;AAErD,gBAAQ,IAAI,aAAa;AAAA,MAC7B,CAAC;AAED,iBAAW;AAAA,IACf,CAAC;AAAA,EACL,OAAO;AACH,0BAAa,CAAC,GAAG,OAAO,EAAE,QAAQ,CAAC,kBAAkB;AAEjD,cAAQ,IAAI,aAAa;AAAA,IAC7B,CAAC;AAAA,EACL;AAIA,UAAQ,IAAI;AAAA,WAAcA,QAAO,YAAY,OAAO,cAAc,MAAM,CAAC,UAAU,cAAc,WAAW,IAAI,UAAU;AAAA,CAAa;AAC3I;AAEA,IAAO,uBAAQ;","names":["colors","join","process","process","fs","join","process","colors"]}
|
|
1
|
+
{"version":3,"sources":["../../../src/next/cli/list/list-command.ts","../../../src/next/cli/list/api-route-file-parser.ts","../../../src/next/cli/list/collect-api-route-files.ts","../../../src/next/cli/list/routes-render.ts"],"sourcesContent":["// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\nimport { join } from \"node:path\";\nimport process from \"node:process\";\n\nimport apiRouteFileParser from \"./api-route-file-parser\";\nimport collectApiRouteFiles from \"./collect-api-route-files\";\nimport routesRender from \"./routes-render\";\nimport type { Route } from \"./types\";\n\ntype RenderOptions = {\n includePaths: string[];\n excludePaths: string[];\n methods: string[];\n group: string;\n};\n\nconst groupBy = (list: Route[], keyGetter: (item: Route) => keyof Route): Map<string, Route[]> => {\n const map = new Map();\n\n list.forEach((item) => {\n const key = keyGetter(item);\n const collection = map.get(key);\n\n if (collection) {\n collection.push(item);\n } else {\n map.set(key, [item]);\n }\n });\n\n return map;\n};\n\nconst listCommand = async (\n path: string = \"\",\n options: Partial<\n {\n verbose: boolean;\n } & RenderOptions\n > = {},\n// eslint-disable-next-line radar/cognitive-complexity\n) => {\n const routePath = join(process.cwd(), path);\n\n const apiRouteFiles = await collectApiRouteFiles(routePath, options.verbose || false);\n\n if (apiRouteFiles.length === 0) {\n // eslint-disable-next-line no-console\n console.error(`No API routes found, in \"${routePath}\".`);\n // eslint-disable-next-line unicorn/no-process-exit\n process.exit(1);\n }\n\n let parsedApiRoutes = apiRouteFiles.flatMap((apiRouteFile) => apiRouteFileParser(apiRouteFile, routePath, options.verbose || false));\n\n if (options.includePaths) {\n parsedApiRoutes = options.includePaths.flatMap((ipath) => parsedApiRoutes.filter((route) => route.path.startsWith(ipath)));\n }\n\n if (options.excludePaths) {\n parsedApiRoutes = options.excludePaths.flatMap((epath) => parsedApiRoutes.filter((route) => !route.path.startsWith(epath)));\n }\n\n if (options.group === undefined) {\n routesRender([], options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n } else {\n const groupedMap = groupBy(parsedApiRoutes, (route) => {\n if (options.group === \"path\") {\n return route.path.replace(\"/pages\", \"\").split(\"/\")[1];\n }\n\n return route.tags[0] || \"unsorted\";\n });\n\n let counter = 0;\n\n groupedMap.forEach((routes, key) => {\n if (counter > 0) {\n // eslint-disable-next-line no-console\n console.log();\n }\n\n const dotsCount = (process.stdout.columns - 16 - key.length) / 2;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\" \").join(\"\") : \"\";\n // eslint-disable-next-line no-console\n console.log(dots + colors.bold.underline(key));\n\n routesRender(routes, options).forEach((renderedRoute) => {\n // eslint-disable-next-line no-console\n console.log(renderedRoute);\n });\n\n counter += 1;\n });\n }\n // });\n\n // eslint-disable-next-line no-console\n console.log(`\\n Listed ${colors.greenBright(String(apiRouteFiles.length))} HTTP ${apiRouteFiles.length === 1 ? \"route\" : \"routes\"}.\\n`);\n};\n\nexport default listCommand;\n","import type { OpenApiObject } from \"@visulima/jsdoc-open-api\";\nimport { jsDocumentCommentsToOpenApi, parseFile, swaggerJsDocumentCommentsToOpenApi } from \"@visulima/jsdoc-open-api\";\nimport fs from \"node:fs\";\nimport process from \"node:process\";\n\nimport type { Route } from \"./types\";\n\nconst extensionRegex = /\\.(js|ts|mjs|cjs)$/;\n\nconst apiRouteFileParser = (apiRouteFile: string, cwdPath: string, verbose: boolean = false): Route[] => {\n let specs: OpenApiObject[] = [];\n\n const parsedJsDocumentFile = parseFile(apiRouteFile, jsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedJsDocumentFile.map((item) => item.spec)];\n\n const parsedSwaggerJsDocumentFile = parseFile(apiRouteFile, swaggerJsDocumentCommentsToOpenApi, verbose);\n\n specs = [...specs, ...parsedSwaggerJsDocumentFile.map((item) => item.spec)];\n\n const routes: Route[] = [];\n\n if (specs.length === 0) {\n const apiRouteFileContent = fs.readFileSync(apiRouteFile, \"utf8\");\n\n apiRouteFileContent.split(/\\r?\\n/).forEach((line) => {\n const match = line.match(/[=aces|]+\\s[\"'|](GET|POST|PUT|PATCH|HEAD|DELETE|OPTIONS)[\"'|]/);\n\n if (match) {\n let [, method] = match;\n\n if (method === \"GET\") {\n method = \"GET|HEAD\";\n }\n\n routes.push({\n method: method as string,\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n });\n\n if (routes.length === 0) {\n routes.push({\n method: \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\",\n path: apiRouteFile.replace(cwdPath, \"\").replace(extensionRegex, \"\"),\n tags: [],\n file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n }\n\n return routes;\n }\n\n specs.forEach((spec) => {\n const paths = Object.entries(spec.paths);\n\n paths.forEach(([path, pathSpec]) => {\n const methods = Object.entries(pathSpec);\n\n methods.forEach(([method, methodSpec]) => {\n routes.push({\n path, method: method.toUpperCase(), tags: methodSpec.tags, file: apiRouteFile.replace(`${process.cwd()}/`, \"\"),\n });\n });\n });\n });\n\n return routes;\n};\n\nexport default apiRouteFileParser;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport { collect } from \"@visulima/readdir\";\nimport fs from \"node:fs\";\nimport { join } from \"node:path\";\n\nconst isDirectory = (path: string): boolean => {\n try {\n return fs.statSync(path).isDirectory();\n } catch {\n return false;\n }\n};\n\nconst collectApiRouteFiles = async (path: string = \"\", verbose: boolean = false): Promise<string[]> => {\n let apiFolderPath = join(path, \"pages/api\");\n\n // src/pages will be ignored if pages is present in the root directory\n if (!isDirectory(apiFolderPath)) {\n apiFolderPath = join(path, \"src/pages/api\");\n\n if (!isDirectory(apiFolderPath)) {\n return [];\n }\n }\n\n return collect(apiFolderPath, {\n extensions: [\".js\", \".cjs\", \".mjs\", \".ts\"],\n includeDirs: false,\n minimatchOptions: {\n match: {\n debug: verbose,\n matchBase: true,\n },\n skip: {\n debug: verbose,\n matchBase: true,\n },\n },\n });\n};\n\nexport default collectApiRouteFiles;\n","// eslint-disable-next-line import/no-extraneous-dependencies\nimport colors from \"chalk\";\n\nimport type { Route } from \"./types\";\n\nconst renderRoute = (method: string, routePath: string): string => {\n const colorMap = {\n GET: colors.blue,\n POST: colors.yellow,\n PATCH: colors.yellow,\n PUT: colors.yellow,\n DELETE: colors.redBright,\n OPTIONS: colors.hex(\"#6C7280\"),\n ANY: colors.redBright,\n HEAD: colors.hex(\"#6C7280\"),\n };\n\n let methodText: string;\n\n if (method === \"GET|HEAD\") {\n methodText = `${colors.blue(\"GET\")}${colors.grey(\"|HEAD\")}`;\n } else {\n const coloredMethod = colorMap[method as keyof typeof colorMap](method);\n\n methodText = method === \"GET\" ? `${coloredMethod}${colors.grey(\"|HEAD\")}` : coloredMethod;\n }\n\n const spacesCount = method === \"GET\" ? 6 : 14 - method.length;\n const spaces = Array.from({ length: spacesCount }).fill(\" \").join(\"\");\n\n const dotsCount = process.stdout.columns - 16 - routePath.length - 4;\n const dots = dotsCount > 0 ? Array.from({ length: dotsCount }).fill(\".\").join(\"\") : \"\";\n\n const routeText = routePath\n .split(\"/\")\n .map((segment) => {\n const isDynamicSegment = [\"[\", \":\"].includes(segment[0] || \"\");\n\n return isDynamicSegment ? colors.yellowBright(segment) : segment;\n })\n .join(\"/\");\n\n return ` ${methodText}${spaces}${routeText}${colors.grey(dots)}`;\n};\n\nconst routesRender = (routesMap: Route[], options: { methods?: string[]; } = {}) => routesMap\n .map((route) => {\n if (options.methods && options.methods.includes(route.method)) {\n return;\n }\n\n if (route.method === \"GET|POST|PUT|PATCH|DELETE|HEAD|OPTIONS\") {\n // eslint-disable-next-line no-param-reassign\n route.method = \"ANY\";\n }\n\n // eslint-disable-next-line consistent-return\n return renderRoute(route.method, route.path.replace(\"/pages\", \"\"));\n })\n .filter(Boolean);\n\nexport default routesRender;\n"],"mappings":";;;AACA,OAAOA,aAAY;AACnB,SAAS,QAAAC,aAAY;AACrB,OAAOC,cAAa;;;ACFpB,SAAS,6BAA6B,WAAW,0CAA0C;AAC3F,OAAO,QAAQ;AACf,OAAOC,cAAa;AAIpB,IAAM,iBAAiB;AAEvB,IAAM,qBAAqB,CAAC,cAAsB,SAAiB,UAAmB,UAAmB;AACrG,MAAI,QAAyB,CAAC;AAE9B,QAAM,uBAAuB,UAAU,cAAc,6BAA6B,OAAO;AAEzF,UAAQ,CAAC,GAAG,OAAO,GAAG,qBAAqB,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC;AAEnE,QAAM,8BAA8B,UAAU,cAAc,oCAAoC,OAAO;AAEvG,UAAQ,CAAC,GAAG,OAAO,GAAG,4BAA4B,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC;AAE1E,QAAM,SAAkB,CAAC;AAEzB,MAAI,MAAM,WAAW,GAAG;AACpB,UAAM,sBAAsB,GAAG,aAAa,cAAc,MAAM;AAEhE,wBAAoB,MAAM,OAAO,EAAE,QAAQ,CAAC,SAAS;AACjD,YAAM,QAAQ,KAAK,MAAM,+DAA+D;AAExF,UAAI,OAAO;AACP,YAAI,CAAC,EAAE,MAAM,IAAI;AAEjB,YAAI,WAAW,OAAO;AAClB,mBAAS;AAAA,QACb;AAEA,eAAO,KAAK;AAAA,UACR;AAAA,UACA,MAAM,aAAa,QAAQ,SAAS,EAAE,EAAE,QAAQ,gBAAgB,EAAE;AAAA,UAClE,MAAM,CAAC;AAAA,UACP,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,QACtD,CAAC;AAAA,MACL;AAAA,IACJ,CAAC;AAED,QAAI,OAAO,WAAW,GAAG;AACrB,aAAO,KAAK;AAAA,QACR,QAAQ;AAAA,QACR,MAAM,aAAa,QAAQ,SAAS,EAAE,EAAE,QAAQ,gBAAgB,EAAE;AAAA,QAClE,MAAM,CAAC;AAAA,QACP,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,MACtD,CAAC;AAAA,IACL;AAEA,WAAO;AAAA,EACX;AAEA,QAAM,QAAQ,CAAC,SAAS;AACpB,UAAM,QAAQ,OAAO,QAAQ,KAAK,KAAK;AAEvC,UAAM,QAAQ,CAAC,CAAC,MAAM,QAAQ,MAAM;AAChC,YAAM,UAAU,OAAO,QAAQ,QAAQ;AAEvC,cAAQ,QAAQ,CAAC,CAAC,QAAQ,UAAU,MAAM;AACtC,eAAO,KAAK;AAAA,UACR;AAAA,UAAM,QAAQ,OAAO,YAAY;AAAA,UAAG,MAAM,WAAW;AAAA,UAAM,MAAM,aAAa,QAAQ,GAAGA,SAAQ,IAAI,MAAM,EAAE;AAAA,QACjH,CAAC;AAAA,MACL,CAAC;AAAA,IACL,CAAC;AAAA,EACL,CAAC;AAED,SAAO;AACX;AAEA,IAAO,gCAAQ;;;ACxEf,SAAS,eAAe;AACxB,OAAOC,SAAQ;AACf,SAAS,YAAY;AAErB,IAAM,cAAc,CAAC,SAA0B;AAC3C,MAAI;AACA,WAAOA,IAAG,SAAS,IAAI,EAAE,YAAY;AAAA,EACzC,QAAE;AACE,WAAO;AAAA,EACX;AACJ;AAEA,IAAM,uBAAuB,OAAO,OAAe,IAAI,UAAmB,UAA6B;AACnG,MAAI,gBAAgB,KAAK,MAAM,WAAW;AAG1C,MAAI,CAAC,YAAY,aAAa,GAAG;AAC7B,oBAAgB,KAAK,MAAM,eAAe;AAE1C,QAAI,CAAC,YAAY,aAAa,GAAG;AAC7B,aAAO,CAAC;AAAA,IACZ;AAAA,EACJ;AAEA,SAAO,QAAQ,eAAe;AAAA,IAC1B,YAAY,CAAC,OAAO,QAAQ,QAAQ,KAAK;AAAA,IACzC,aAAa;AAAA,IACb,kBAAkB;AAAA,MACd,OAAO;AAAA,QACH,OAAO;AAAA,QACP,WAAW;AAAA,MACf;AAAA,MACA,MAAM;AAAA,QACF,OAAO;AAAA,QACP,WAAW;AAAA,MACf;AAAA,IACJ;AAAA,EACJ,CAAC;AACL;AAEA,IAAO,kCAAQ;;;ACxCf,OAAO,YAAY;AAInB,IAAM,cAAc,CAAC,QAAgB,cAA8B;AAC/D,QAAM,WAAW;AAAA,IACb,KAAK,OAAO;AAAA,IACZ,MAAM,OAAO;AAAA,IACb,OAAO,OAAO;AAAA,IACd,KAAK,OAAO;AAAA,IACZ,QAAQ,OAAO;AAAA,IACf,SAAS,OAAO,IAAI,SAAS;AAAA,IAC7B,KAAK,OAAO;AAAA,IACZ,MAAM,OAAO,IAAI,SAAS;AAAA,EAC9B;AAEA,MAAI;AAEJ,MAAI,WAAW,YAAY;AACvB,iBAAa,GAAG,OAAO,KAAK,KAAK,IAAI,OAAO,KAAK,OAAO;AAAA,EAC5D,OAAO;AACH,UAAM,gBAAgB,SAAS,QAAiC,MAAM;AAEtE,iBAAa,WAAW,QAAQ,GAAG,gBAAgB,OAAO,KAAK,OAAO,MAAM;AAAA,EAChF;AAEA,QAAM,cAAc,WAAW,QAAQ,IAAI,KAAK,OAAO;AACvD,QAAM,SAAS,MAAM,KAAK,EAAE,QAAQ,YAAY,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE;AAEpE,QAAM,YAAY,QAAQ,OAAO,UAAU,KAAK,UAAU,SAAS;AACnE,QAAM,OAAO,YAAY,IAAI,MAAM,KAAK,EAAE,QAAQ,UAAU,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE,IAAI;AAEpF,QAAM,YAAY,UACb,MAAM,GAAG,EACT,IAAI,CAAC,YAAY;AACd,UAAM,mBAAmB,CAAC,KAAK,GAAG,EAAE,SAAS,QAAQ,MAAM,EAAE;AAE7D,WAAO,mBAAmB,OAAO,aAAa,OAAO,IAAI;AAAA,EAC7D,CAAC,EACA,KAAK,GAAG;AAEb,SAAO,KAAK,aAAa,SAAS,YAAY,OAAO,KAAK,IAAI;AAClE;AAEA,IAAM,eAAe,CAAC,WAAoB,UAAmC,CAAC,MAAM,UAC/E,IAAI,CAAC,UAAU;AACZ,MAAI,QAAQ,WAAW,QAAQ,QAAQ,SAAS,MAAM,MAAM,GAAG;AAC3D;AAAA,EACJ;AAEA,MAAI,MAAM,WAAW,0CAA0C;AAE3D,UAAM,SAAS;AAAA,EACnB;AAGA,SAAO,YAAY,MAAM,QAAQ,MAAM,KAAK,QAAQ,UAAU,EAAE,CAAC;AACrE,CAAC,EACA,OAAO,OAAO;AAEnB,IAAO,wBAAQ;;;AH5Cf,IAAM,UAAU,CAAC,MAAe,cAAkE;AAC9F,QAAM,MAAM,oBAAI,IAAI;AAEpB,OAAK,QAAQ,CAAC,SAAS;AACnB,UAAM,MAAM,UAAU,IAAI;AAC1B,UAAM,aAAa,IAAI,IAAI,GAAG;AAE9B,QAAI,YAAY;AACZ,iBAAW,KAAK,IAAI;AAAA,IACxB,OAAO;AACH,UAAI,IAAI,KAAK,CAAC,IAAI,CAAC;AAAA,IACvB;AAAA,EACJ,CAAC;AAED,SAAO;AACX;AAEA,IAAM,cAAc,OAChB,OAAe,IACf,UAII,CAAC,MAEJ;AACD,QAAM,YAAYC,MAAKC,SAAQ,IAAI,GAAG,IAAI;AAE1C,QAAM,gBAAgB,MAAM,gCAAqB,WAAW,QAAQ,WAAW,KAAK;AAEpF,MAAI,cAAc,WAAW,GAAG;AAE5B,YAAQ,MAAM,4BAA4B,aAAa;AAEvD,IAAAA,SAAQ,KAAK,CAAC;AAAA,EAClB;AAEA,MAAI,kBAAkB,cAAc,QAAQ,CAAC,iBAAiB,8BAAmB,cAAc,WAAW,QAAQ,WAAW,KAAK,CAAC;AAEnI,MAAI,QAAQ,cAAc;AACtB,sBAAkB,QAAQ,aAAa,QAAQ,CAAC,UAAU,gBAAgB,OAAO,CAAC,UAAU,MAAM,KAAK,WAAW,KAAK,CAAC,CAAC;AAAA,EAC7H;AAEA,MAAI,QAAQ,cAAc;AACtB,sBAAkB,QAAQ,aAAa,QAAQ,CAAC,UAAU,gBAAgB,OAAO,CAAC,UAAU,CAAC,MAAM,KAAK,WAAW,KAAK,CAAC,CAAC;AAAA,EAC9H;AAEA,MAAI,QAAQ,UAAU,QAAW;AAC7B,0BAAa,CAAC,GAAG,OAAO,EAAE,QAAQ,CAAC,kBAAkB;AAEjD,cAAQ,IAAI,aAAa;AAAA,IAC7B,CAAC;AAAA,EACL,OAAO;AACH,UAAM,aAAa,QAAQ,iBAAiB,CAAC,UAAU;AACnD,UAAI,QAAQ,UAAU,QAAQ;AAC1B,eAAO,MAAM,KAAK,QAAQ,UAAU,EAAE,EAAE,MAAM,GAAG,EAAE;AAAA,MACvD;AAEA,aAAO,MAAM,KAAK,MAAM;AAAA,IAC5B,CAAC;AAED,QAAI,UAAU;AAEd,eAAW,QAAQ,CAAC,QAAQ,QAAQ;AAChC,UAAI,UAAU,GAAG;AAEb,gBAAQ,IAAI;AAAA,MAChB;AAEA,YAAM,aAAaA,SAAQ,OAAO,UAAU,KAAK,IAAI,UAAU;AAC/D,YAAM,OAAO,YAAY,IAAI,MAAM,KAAK,EAAE,QAAQ,UAAU,CAAC,EAAE,KAAK,GAAG,EAAE,KAAK,EAAE,IAAI;AAEpF,cAAQ,IAAI,OAAOC,QAAO,KAAK,UAAU,GAAG,CAAC;AAE7C,4BAAa,QAAQ,OAAO,EAAE,QAAQ,CAAC,kBAAkB;AAErD,gBAAQ,IAAI,aAAa;AAAA,MAC7B,CAAC;AAED,iBAAW;AAAA,IACf,CAAC;AAAA,EACL;AAIA,UAAQ,IAAI;AAAA,WAAcA,QAAO,YAAY,OAAO,cAAc,MAAM,CAAC,UAAU,cAAc,WAAW,IAAI,UAAU;AAAA,CAAa;AAC3I;AAEA,IAAO,uBAAQ;","names":["colors","join","process","process","fs","join","process","colors"]}
|
|
@@ -2,11 +2,11 @@
|
|
|
2
2
|
|
|
3
3
|
|
|
4
4
|
|
|
5
|
-
var
|
|
5
|
+
var _chunkF7RHRCUQjs = require('../chunk-F7RHRCUQ.js');
|
|
6
6
|
require('../chunk-JC4IRQUL.js');
|
|
7
7
|
|
|
8
8
|
|
|
9
9
|
|
|
10
10
|
|
|
11
|
-
exports.RedocPage =
|
|
11
|
+
exports.RedocPage = _chunkF7RHRCUQjs.redoc_default; exports.SwaggerPage = _chunkF7RHRCUQjs.swagger_default; exports.getSwaggerStaticProps = _chunkF7RHRCUQjs.get_static_properties_swagger_default;
|
|
12
12
|
//# sourceMappingURL=index-browser.js.map
|