{"id":13527380,"url":"https://github.com/vercel/serve-handler","last_synced_at":"2026-01-15T22:20:33.546Z","repository":{"id":38304779,"uuid":"133409152","full_name":"vercel/serve-handler","owner":"vercel","description":"The foundation of `serve`","archived":false,"fork":false,"pushed_at":"2024-10-15T13:55:41.000Z","size":639,"stargazers_count":613,"open_issues_count":82,"forks_count":100,"subscribers_count":62,"default_branch":"main","last_synced_at":"2026-01-13T19:24:19.806Z","etag":null,"topics":["handler","middleware","server","single-page-applications","static"],"latest_commit_sha":null,"homepage":"https://npmjs.com/serve-handler","language":"JavaScript","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/vercel.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":null,"funding":null,"license":"LICENSE","code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null}},"created_at":"2018-05-14T19:08:11.000Z","updated_at":"2026-01-09T11:31:07.000Z","dependencies_parsed_at":"2023-02-05T02:15:52.177Z","dependency_job_id":"cfe09210-c53e-4abd-ab74-3283c65b87a6","html_url":"https://github.com/vercel/serve-handler","commit_stats":{"total_commits":211,"total_committers":24,"mean_commits":8.791666666666666,"dds":"0.20853080568720384","last_synced_commit":"17f3f7e70a7f54b4f82c9c18376969fa15810c6e"},"previous_names":["zeit/serve-handler"],"tags_count":66,"template":false,"template_full_name":null,"purl":"pkg:github/vercel/serve-handler","repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/vercel%2Fserve-handler","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/vercel%2Fserve-handler/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/vercel%2Fserve-handler/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/vercel%2Fserve-handler/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/vercel","download_url":"https://codeload.github.com/vercel/serve-handler/tar.gz/refs/heads/main","sbom_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/vercel%2Fserve-handler/sbom","scorecard":{"id":918761,"data":{"date":"2025-08-11","repo":{"name":"github.com/vercel/serve-handler","commit":"329bcba871abed617d4b32cfefd79b5ddc35e497"},"scorecard":{"version":"v5.2.1-40-gf6ed084d","commit":"f6ed084d17c9236477efd66e5b258b9d4cc7b389"},"score":4.1,"checks":[{"name":"Code-Review","score":2,"reason":"Found 7/27 approved changesets -- score normalized to 2","details":null,"documentation":{"short":"Determines if the project requires human code review before pull requests (aka merge requests) are merged.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#code-review"}},{"name":"Maintained","score":0,"reason":"0 commit(s) and 0 issue activity found in the last 90 days -- score normalized to 0","details":null,"documentation":{"short":"Determines if the project is \"actively maintained\".","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#maintained"}},{"name":"Pinned-Dependencies","score":0,"reason":"dependency not pinned by hash detected -- score normalized to 0","details":["Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/publish.yaml:12: update your workflow using https://app.stepsecurity.io/secureworkflow/vercel/serve-handler/publish.yaml/main?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/publish.yaml:13: update your workflow using https://app.stepsecurity.io/secureworkflow/vercel/serve-handler/publish.yaml/main?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/tests.yaml:14: update your workflow using https://app.stepsecurity.io/secureworkflow/vercel/serve-handler/tests.yaml/main?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/tests.yaml:17: update your workflow using https://app.stepsecurity.io/secureworkflow/vercel/serve-handler/tests.yaml/main?enable=pin","Warn: third-party GitHubAction not pinned by hash: .github/workflows/tests.yaml:27: update your workflow using https://app.stepsecurity.io/secureworkflow/vercel/serve-handler/tests.yaml/main?enable=pin","Info:   0 out of   4 GitHub-owned GitHubAction dependencies pinned","Info:   0 out of   1 third-party GitHubAction dependencies pinned"],"documentation":{"short":"Determines if the project has declared and pinned the dependencies of its build process.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#pinned-dependencies"}},{"name":"Binary-Artifacts","score":10,"reason":"no binaries found in the repo","details":null,"documentation":{"short":"Determines if the project has generated executable (binary) artifacts in the source repository.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#binary-artifacts"}},{"name":"Token-Permissions","score":0,"reason":"detected GitHub workflow tokens with excessive permissions","details":["Warn: no topLevel permission defined: .github/workflows/publish.yaml:1","Warn: no topLevel permission defined: .github/workflows/tests.yaml:1","Info: no jobLevel write permissions found"],"documentation":{"short":"Determines if the project's workflows follow the principle of least privilege.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#token-permissions"}},{"name":"CII-Best-Practices","score":0,"reason":"no effort to earn an OpenSSF best practices badge detected","details":null,"documentation":{"short":"Determines if the project has an OpenSSF (formerly CII) Best Practices Badge.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#cii-best-practices"}},{"name":"Dangerous-Workflow","score":10,"reason":"no dangerous workflow patterns detected","details":null,"documentation":{"short":"Determines if the project's GitHub Action workflows avoid dangerous patterns.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#dangerous-workflow"}},{"name":"Fuzzing","score":0,"reason":"project is not fuzzed","details":["Warn: no fuzzer integrations found"],"documentation":{"short":"Determines if the project uses fuzzing.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#fuzzing"}},{"name":"License","score":10,"reason":"license file detected","details":["Info: project has a license file: LICENSE:0","Info: FSF or OSI recognized license: MIT License: LICENSE:0"],"documentation":{"short":"Determines if the project has defined a license.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#license"}},{"name":"Packaging","score":10,"reason":"packaging workflow detected","details":["Info: Project packages its releases by way of GitHub Actions.: .github/workflows/publish.yaml:9"],"documentation":{"short":"Determines if the project is published as a package that others can easily download, install, easily update, and uninstall.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#packaging"}},{"name":"Signed-Releases","score":-1,"reason":"no releases found","details":null,"documentation":{"short":"Determines if the project cryptographically signs release artifacts.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#signed-releases"}},{"name":"Branch-Protection","score":-1,"reason":"internal error: error during branchesHandler.setup: internal error: githubv4.Query: Resource not accessible by integration","details":null,"documentation":{"short":"Determines if the default and release branches are protected with GitHub's branch protection settings.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#branch-protection"}},{"name":"Security-Policy","score":10,"reason":"security policy file detected","details":["Info: security policy file detected: github.com/vercel/.github/SECURITY.md:1","Info: Found linked content: github.com/vercel/.github/SECURITY.md:1","Info: Found disclosure, vulnerability, and/or timelines in security policy: github.com/vercel/.github/SECURITY.md:1","Info: Found text in security policy: github.com/vercel/.github/SECURITY.md:1"],"documentation":{"short":"Determines if the project has published a security policy.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#security-policy"}},{"name":"SAST","score":0,"reason":"SAST tool is not run on all commits -- score normalized to 0","details":["Warn: 0 commits out of 20 are checked with a SAST tool"],"documentation":{"short":"Determines if the project uses static code analysis.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#sast"}},{"name":"Vulnerabilities","score":0,"reason":"19 existing vulnerabilities detected","details":["Warn: Project is vulnerable to: GHSA-968p-4wvh-cqc8","Warn: Project is vulnerable to: GHSA-v88g-cgmw-v5xw","Warn: Project is vulnerable to: GHSA-93q8-gq69-wqmw","Warn: Project is vulnerable to: GHSA-v6h2-p8h4-qcjw","Warn: Project is vulnerable to: GHSA-3xgq-45jj-v275","Warn: Project is vulnerable to: GHSA-gxpj-cx7g-858c","Warn: Project is vulnerable to: GHSA-fjxv-7rqg-78g4","Warn: Project is vulnerable to: GHSA-ww39-953v-wcq6","Warn: Project is vulnerable to: GHSA-896r-f27r-55mw","Warn: Project is vulnerable to: GHSA-29mw-wpgm-hmr9","Warn: Project is vulnerable to: GHSA-35jh-r3h4-6jhm","Warn: Project is vulnerable to: GHSA-xvch-5gv4-984h","Warn: Project is vulnerable to: GHSA-r683-j2x4-v87g","Warn: Project is vulnerable to: GHSA-hrpp-h998-j3pp","Warn: Project is vulnerable to: GHSA-p8p7-x288-28g6","Warn: Project is vulnerable to: GHSA-c2qf-rxjj-qqgw","Warn: Project is vulnerable to: GHSA-52f5-9888-hmc6","Warn: Project is vulnerable to: GHSA-72xf-g2v4-qvf3","Warn: Project is vulnerable to: GHSA-j8xg-fqg3-53r7"],"documentation":{"short":"Determines if the project has open, known unfixed vulnerabilities.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#vulnerabilities"}}]},"last_synced_at":"2025-08-24T23:18:33.500Z","repository_id":38304779,"created_at":"2025-08-24T23:18:33.501Z","updated_at":"2025-08-24T23:18:33.501Z"},"host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":286080680,"owners_count":28472626,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2026-01-15T22:13:38.078Z","status":"ssl_error","status_checked_at":"2026-01-15T22:12:11.737Z","response_time":62,"last_error":"SSL_connect returned=1 errno=0 peeraddr=140.82.121.6:443 state=error: unexpected eof while reading","robots_txt_status":"success","robots_txt_updated_at":"2025-07-24T06:49:26.215Z","robots_txt_url":"https://github.com/robots.txt","online":false,"can_crawl_api":true,"host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["handler","middleware","server","single-page-applications","static"],"created_at":"2024-08-01T06:01:46.810Z","updated_at":"2026-01-15T22:20:33.532Z","avatar_url":"https://github.com/vercel.png","language":"JavaScript","readme":"# serve-handler\n\n[![Tests](https://github.com/vercel/serve-handler/actions/workflows/tests.yaml/badge.svg)](https://github.com/vercel/serve-handler/actions/workflows/tests.yaml)\n[![codecov](https://codecov.io/gh/vercel/serve-handler/branch/main/graph/badge.svg)](https://codecov.io/gh/vercel/serve-handler)\n[![install size](https://packagephobia.now.sh/badge?p=serve-handler)](https://packagephobia.now.sh/result?p=serve-handler)\n\nThis package represents the core of [serve](https://github.com/vercel/serve). It can be plugged into any HTTP server and is responsible for routing requests and handling responses.\n\nIn order to customize the default behaviour, you can also pass custom routing rules, provide your own methods for interacting with the file system and much more.\n\n## Usage\n\nGet started by installing the package using [yarn](https://yarnpkg.com/lang/en/):\n\n```sh\nyarn add serve-handler\n```\n\nYou can also use [npm](https://www.npmjs.com/) instead, if you'd like:\n\n```sh\nnpm install serve-handler\n```\n\nNext, add it to your HTTP server. Here's an example using [micro](https://github.com/vercel/micro):\n\n```js\nconst handler = require('serve-handler');\n\nmodule.exports = async (request, response) =\u003e {\n  await handler(request, response);\n};\n```\n\nThat's it! :tada:\n\n## Options\n\nIf you want to customize the package's default behaviour, you can use the third argument of the function call to pass any of the configuration options listed below. Here's an example:\n\n```js\nawait handler(request, response, {\n  cleanUrls: true\n});\n```\n\nYou can use any of the following options:\n\n| Property                                             | Description                                                           |\n|------------------------------------------------------|-----------------------------------------------------------------------|\n| [`public`](#public-string)                           | Set a sub directory to be served                                      |\n| [`cleanUrls`](#cleanurls-booleanarray)               | Have the `.html` extension stripped from paths                        |\n| [`rewrites`](#rewrites-array)                        | Rewrite paths to different paths                                      |\n| [`redirects`](#redirects-array)                      | Forward paths to different paths or external URLs                     |\n| [`headers`](#headers-array)                          | Set custom headers for specific paths                                 |\n| [`directoryListing`](#directorylisting-booleanarray) | Disable directory listing or restrict it to certain paths             |\n| [`unlisted`](#unlisted-array)                        | Exclude paths from the directory listing                              |\n| [`trailingSlash`](#trailingslash-boolean)            | Remove or add trailing slashes to all paths                           |\n| [`renderSingle`](#rendersingle-boolean)              | If a directory only contains one file, render it                      |\n| [`symlinks`](#symlinks-boolean)                      | Resolve symlinks instead of rendering a 404 error                     |\n| [`etag`](#etag-boolean)                              | Calculate a strong `ETag` response header, instead of `Last-Modified` |\n\n### public (String)\n\nBy default, the current working directory will be served. If you only want to serve a specific path, you can use this options to pass an absolute path or a custom directory to be served relative to the current working directory.\n\nFor example, if serving a [Jekyll](https://jekyllrb.com/) app, it would look like this:\n\n```json\n{\n  \"public\": \"_site\"\n}\n```\n\nUsing absolute path:\n\n```json\n{\n  \"public\": \"/path/to/your/_site\"\n}\n```\n\n**NOTE:** The path cannot contain globs or regular expressions.\n\n### cleanUrls (Boolean|Array)\n\nBy default, all `.html` files can be accessed without their extension.\n\nIf one of these extensions is used at the end of a filename, it will automatically perform a redirect with status code [301](https://en.wikipedia.org/wiki/HTTP_301) to the same path, but with the extension dropped.\n\nYou can disable the feature like follows:\n\n```json\n{\n  \"cleanUrls\": false\n}\n```\n\nHowever, you can also restrict it to certain paths:\n\n```json\n{\n  \"cleanUrls\": [\n    \"/app/**\",\n    \"/!components/**\"\n  ]\n}\n```\n\n**NOTE:** The paths can only contain globs that are matched using [minimatch](https://github.com/isaacs/minimatch).\n\n### rewrites (Array)\n\nIf you want your visitors to receive a response under a certain path, but actually serve a completely different one behind the curtains, this option is what you need.\n\nIt's perfect for [single page applications](https://en.wikipedia.org/wiki/Single-page_application) (SPAs), for example:\n\n```json\n{\n  \"rewrites\": [\n    { \"source\": \"app/**\", \"destination\": \"/index.html\" },\n    { \"source\": \"projects/*/edit\", \"destination\": \"/edit-project.html\" }\n  ]\n}\n```\n\nYou can also use so-called \"routing segments\" as follows:\n\n```json\n{\n  \"rewrites\": [\n    { \"source\": \"/projects/:id/edit\", \"destination\": \"/edit-project-:id.html\" },\n  ]\n}\n```\n\nNow, if a visitor accesses `/projects/123/edit`, it will respond with the file `/edit-project-123.html`.\n\n**NOTE:** The paths can contain globs (matched using [minimatch](https://github.com/isaacs/minimatch)) or regular expressions (match using [path-to-regexp](https://github.com/pillarjs/path-to-regexp)).\n\n### redirects (Array)\n\nIn order to redirect visits to a certain path to a different one (or even an external URL), you can use this option:\n\n```json\n{\n  \"redirects\": [\n    { \"source\": \"/from\", \"destination\": \"/to\" },\n    { \"source\": \"/old-pages/**\", \"destination\": \"/home\" }\n  ]\n}\n```\n\nBy default, all of them are performed with the status code [301](https://en.wikipedia.org/wiki/HTTP_301), but this behavior can be adjusted by setting the `type` property directly on the object (see below).\n\nJust like with [rewrites](#rewrites-array), you can also use routing segments:\n\n```json\n{\n  \"redirects\": [\n    { \"source\": \"/old-docs/:id\", \"destination\": \"/new-docs/:id\" },\n    { \"source\": \"/old\", \"destination\": \"/new\", \"type\": 302 }\n  ]\n}\n```\n\nIn the example above, `/old-docs/12` would be forwarded to `/new-docs/12` with status code [301](https://en.wikipedia.org/wiki/HTTP_301). In addition `/old` would be forwarded to `/new` with status code [302](https://en.wikipedia.org/wiki/HTTP_302).\n\n**NOTE:** The paths can contain globs (matched using [minimatch](https://github.com/isaacs/minimatch)) or regular expressions (match using [path-to-regexp](https://github.com/pillarjs/path-to-regexp)).\n\n### headers (Array)\n\nAllows you to set custom headers (and overwrite the default ones) for certain paths:\n\n```json\n{\n  \"headers\": [\n    {\n      \"source\" : \"**/*.@(jpg|jpeg|gif|png)\",\n      \"headers\" : [{\n        \"key\" : \"Cache-Control\",\n        \"value\" : \"max-age=7200\"\n      }]\n    }, {\n      \"source\" : \"404.html\",\n      \"headers\" : [{\n        \"key\" : \"Cache-Control\",\n        \"value\" : \"max-age=300\"\n      }]\n    }\n  ]\n}\n```\n\nIf you define the `ETag` header for a path, the handler will automatically reply with status code `304` for that path if a request comes in with a matching `If-None-Match` header.\n\nIf you set a header `value` to `null` it removes any previous defined header with the same key.\n\n**NOTE:** The paths can only contain globs that are matched using [minimatch](https://github.com/isaacs/minimatch).\n\n### directoryListing (Boolean|Array)\n\nFor paths are not files, but directories, the package will automatically render a good-looking list of all the files and directories contained inside that directory.\n\nIf you'd like to disable this for all paths, set this option to `false`. Furthermore, you can also restrict it to certain directory paths if you want:\n\n```json\n{\n  \"directoryListing\": [\n    \"/assets/**\",\n    \"/!assets/private\"\n  ]\n}\n```\n\n**NOTE:** The paths can only contain globs that are matched using [minimatch](https://github.com/isaacs/minimatch).\n\n### unlisted (Array)\n\nIn certain cases, you might not want a file or directory to appear in the directory listing. In these situations, there are two ways of solving this problem.\n\nEither you disable the directory listing entirely (like shown [here](#directorylisting-booleanarray)), or you exclude certain paths from those listings by adding them all to this config property.\n\n```json\n{\n  \"unlisted\": [\n    \".DS_Store\",\n    \".git\"\n  ]\n}\n```\n\nThe items shown above are excluded from the directory listing by default.\n\n**NOTE:** The paths can only contain globs that are matched using [minimatch](https://github.com/isaacs/minimatch).\n\n### trailingSlash (Boolean)\n\nBy default, the package will try to make assumptions for when to add trailing slashes to your URLs or not. If you want to remove them, set this property to `false` and `true` if you want to force them on all URLs:\n\n```js\n{\n  \"trailingSlash\": true\n}\n```\n\nWith the above config, a request to `/test` would now result in a [301](https://en.wikipedia.org/wiki/HTTP_301) redirect to `/test/`.\n\n### renderSingle (Boolean)\n\nSometimes you might want to have a directory path actually render a file, if the directory only contains one. This is only useful for any files that are not `.html` files (for those, [`cleanUrls`](#cleanurls-booleanarray) is faster).\n\nThis is disabled by default and can be enabled like this:\n\n```js\n{\n  \"renderSingle\": true\n}\n```\n\nAfter that, if you access your directory `/test` (for example), you will see an image being rendered if the directory contains a single image file.\n\n### symlinks (Boolean)\n\nFor security purposes, symlinks are disabled by default. If `serve-handler` encounters a symlink, it will treat it as if it doesn't exist in the first place. In turn, a 404 error is rendered for that path.\n\nHowever, this behavior can easily be adjusted:\n\n```js\n{\n  \"symlinks\": true\n}\n```\n\nOnce this property is set as shown above, all symlinks will automatically be resolved to their targets.\n\n### etag (Boolean)\n\nHTTP response headers will contain a strong [`ETag`][etag] response header, instead of a [`Last-Modified`][last-modified] header. Opt-in because calculating the hash value may be computationally expensive for large files.\n\nSending an `ETag` header is disabled by default and can be enabled like this:\n\n```js\n{\n  \"etag\": true\n}\n```\n\n## Error templates\n\nThe handler will automatically determine the right error format if one occurs and then sends it to the client in that format.\n\nFurthermore, this allows you to not just specifiy an error template for `404` errors, but also for all other errors that can occur (e.g. `400` or `500`).\n\nJust add a `\u003cstatus-code\u003e.html` file to the root directory and you're good.\n\n## Middleware\n\nIf you want to replace the methods the package is using for interacting with the file system and sending responses, you can pass them as the fourth argument to the function call.\n\nThese are the methods used by the package (they can all return a `Promise` or be asynchronous):\n\n```js\nawait handler(request, response, undefined, {\n  lstat(path) {},\n  realpath(path) {},\n  createReadStream(path, config) {}\n  readdir(path) {},\n  sendError(absolutePath, response, acceptsJSON, root, handlers, config, error) {}\n});\n```\n\n**NOTE:** It's important that – for native methods like `createReadStream` – all arguments are passed on to the native call.\n\n## Author\n\nLeo Lamprecht ([@leo](https://x.com/leo)) - [Vercel](https://vercel.com)\n\n\n[etag]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/ETag\n[last-modified]: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Last-Modified\n","funding_links":[],"categories":["JavaScript"],"sub_categories":[],"project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fvercel%2Fserve-handler","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Fvercel%2Fserve-handler","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fvercel%2Fserve-handler/lists"}